首先,在linux中,ID为0的进程通常是调度进程,常常被称为交换进程。该进程是内核的一部分,他并不执行任何磁盘上的程序,因此也被称为系统进程。进程为1的进程通常是init进程,在自举过程结束时由内核调用。该进程的程序文件在UNIX的早期版本中是/etc/init,在较新的版本中是/sbin/init。此进程负责在自举内核后启动一个UNIX系统,init进程决不会终止。它是一个普通的用户进程。但是他是以超级用户特权运行的。
linux中,基本所有的进程都是init进程的子孙进程,init也会接收所有的孤儿进程作为自己的子进程。由init产生子进程,或是由其他进程参数子进程,所使用的函数都是fork();下面小沃就来介绍一下fork函数。
fork()这个函数在unistd.h中,要想调用fork(),需要先包入unistd.h。
fork()函数的完整原型为pid_t fork();
pid_t其实也就只是一个int型。由fork创建的新进程被称为子进程。fork函数被调用一次,但返回两次。两次返回的唯一区别是子进程的返回值是0,而父进程的返回值则是新子进程的进程ID。
子进程和父进程继续执行fork调用之后的指令。子进程是父进程的副本。例如,子进程获得父进程数据空间、堆和栈的副本。子进程的数据空间使用了写时复制技术。这些区域由父子进程共享,而且内核将他们的访问权限改变为只读的。如果父、子进程中的任意一个试图修改这些区域,则内核只为修改区域的那块内存制作一个副本。
下面提供一段使用fork的小示例代码:#include
#include
int glob = 6;
char buf[] = "a write to stdout\n";
int main(void) {
int var;
pid_t pid;
var = 88;
printf(buf);
printf("before fork\n");
if((pid = fork())
printf("fork error\n");
} else if(pid == 0) {//子进程
glob++;
var++;
} else {//父进程
sleep(2);
}
printf("pid = %d, glob = %d, var = %d\n", getpid(), glob, var);
return 0;
}
a write to stdout
before fork
pid = 870, glob = 7, var = 89 #子进程的变量值改变了
pid = 868, glob = 6, var = 88 #父进程的变量值没有改变
一般来说,在fork之后是父进程先执行还是子进程先执行是不确定的。这取决于内核所使用的调度算法。在程序中,父进程使自己休眠2s,以使子进程先执行。