程序的实例被称为进程,有三个用于进程控制的主要函数:fork、exec和waitpid。
接下来会有一段类shell程序的简化实现代码,让我们一起来看看以上3个函数是如何使用的,随后参照代码,会有比较详细的要点解释。
#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/wait.h>
#define MAXLINE 4096
int main(int argc, char* argv[])
{
pid_t pid;
char buf[MAXLINE] = "";
int len = 0;
int status = 0;
printf("%% ");
while (fgets(buf, MAXLINE, stdin))
{
len = strlen(buf);
if (buf[len - 1] == '\n')
{
buf[len - 1] = '\0';
}
if ((pid = fork()) < 0)
{
printf("fork error\n");
exit(1);
}
else if (pid == 0) //child
{
execlp(buf, buf, (char*)0);
printf("couldn't execute: %s\n", buf);
exit(127);
}
//parent
if ((pid = waitpid(pid, &status, 0)) < 0)
{
printf("waitpid error\n");
}
printf("%% ");
}
}
1、用标准I/O函数fgets从标准输入一次读一行,当键入文件结束字符(ctrl+D)作为一行的第一个字符时,fgets返回一个null指针,于是循环退出,进程也就终止。
2、fgets返回的每一行都以换行符终止,后随一个null字符,故用标准C函数strlen计算此字符串长度,并用一个null字节代替换行符。这样做是因为execlp函数要求参数以null而不是换行符结束。
3、调用fork创建一个新的进程,它是调用进程的一个拷贝。我们称调用进程为父进程,新创建的进程为子进程。fork给父进程返回一个非负整数,给子进程返回0。fork被调用一次(被父进程调用),但是返回两次(分别在父进程和子进程中返回)。
4、waitpid可以显示指定需要等待哪个子进程结束。