c 进程和系统调用

这一篇博客讲解进程和系统调用相关的知识

有这样一个场景,我需要输入一串文字,然后把我输入的文字加上一个本地的时间戳 保存在一个文件中,可以初步理解为一个备忘录也行

 #include <stdio.h>
#include <stdlib.h>
#include <time.h> /**
* 获取一个当前时间
*/
char *now() {
time_t t;
time(&t);
return asctime(localtime(&t));
} int main(int argc, const char * argv[]) { char comment[];
char cmd[]; fgets(comment, , stdin);
// sprintf 把内容写进一个变量中
sprintf(cmd, "echo '%s %s' >> reports.log",comment,now());
// 调用系统的方法来执行这段命令
system(cmd);
return ;
}

程序运行的结果是生成了一个文件

c 进程和系统调用

但有时候systemt() 函数也会是不安全的

sprintf(cmd, "echo ''&& ls / && echo' %s' >> reports.log",now());

修改上边的代码,然后会得到这样的结果

c 进程和系统调用

列出了根目录下的内容,因此可以使用命令删除文件或启动病毒

c 进程和系统调用

c 进程和系统调用

c 进程和系统调用

c 进程和系统调用

c 进程和系统调用

c 进程和系统调用

那么现在有一个场景,我们有exec来获取本机的网络配置,在Linux 和 Mac 上 你可以用一个叫/sbin/ifconfig的程序 在window上你用ipconfig

看代码

 #include <stdio.h>
#include <errno.h>
#include <unistd.h>
#include <string.h> int main(int argc, const char * argv[]) { if (execl("/sbin/ifconfig", "/sbin/ifconfig",NULL) == -) { if (execlp("ipconfig", "ipconfig",NULL) == -) { fprintf(stderr, "Can not run ipconfig : %s",strerror(errno));
}
}
return ;
}

下边的代码也是很有意思的,可以充分说明exec函数的作用是停止当前进程 跳到另一个进程的

我们先创建一个叫做coffee.c 的文件,代码如下

 #include <stdio.h>
#include <stdlib.h> int main(int argc, char* argv[]) { char *w = getenv("EXTRA");
if (!w) {
w = getenv("FOOD");
}
if (!w) {
w = argv[argc - ];
} char *c = getenv("EXTRA");
if (!c) {
c = argv[argc - ];
} printf("%s with %s \n", c, w);
return ;
}

该程序的作用是先检测是否有环境变量EXTRA 没有就检测FOOD 如果还没有就给w赋值为最后一个参数值

先检测环境变量EXTRA 如果没有就给c赋值为最后一个参数

下边的代码可以检测我们上边这段代码是不是能够正确运行,首先我们新建一个coffeeTest.c的文件

 #include <stdio.h>
#include <string.h>
#include <errno.h>
#include <unistd.h> int main(int argc, char* argv[]) { char *my_env[] = {"FOOD=Hanbaobao", NULL};
if (execle("./coffee", "./coffee", "abv", NULL, my_env) == -) {
fprintf(stderr, "Can not run process: %s",strerror(errno));
return ;
}
return ;
}
 gcc coffee.c -o coffee
gcc coffeeTest.c -o coffeeTest
./coffeeTest

得到的结果是

abv with Hanbaobao 

假如我们吧coffeeTest.c中的代码改成这样呢

  if (execl("./coffee", "./coffee", "abv", NULL) == -) {
fprintf(stderr, "Can not run process: %s",strerror(errno));
return ;
}

不出意外结果就是

abv with abv 

好了,我们已经了解exec函数的使用方法了,其实exec函数算是程序中的最后一行代码了,只要调用它之后,程序立即停止运行,原理就是停止当前进程开启另外一个进程

那么如果我们还想要后边的代码继续执行呢

先看个例子

 #include <stdio.h>
#include <string.h>
#include <errno.h>
#include <unistd.h> int main(int argc, char* argv[]) { char *my_env[] = {"FOOD=Hanbaobao", NULL}; for (int i = ; i < ; i++) { pid_t pid = fork(); if (pid == -) {
fprintf(stderr, "Can not fork process: %s",strerror(errno));
return ;
} if (!pid) { if (execle("./coffee", "./coffee", "abv", NULL, my_env) == -) {
fprintf(stderr, "Can not run process: %s",strerror(errno));
return ;
}
} }
return ;
}

如果不加fork() 上边的程序只会打印一次结果,当循环中第一次调用exec的时候,后边的代码就不会再调用了

fork 是什么,看下边的解释

c 进程和系统调用

c 进程和系统调用

c 进程和系统调用

c 进程和系统调用

c 进程和系统调用

上一篇:[转]ASP.NET MVC 入门8、ModelState与数据验证


下一篇:struts 文件上传