1、进程标识
每个进程都有一个非负整型的唯一进程ID。
有某些专用的进程:
进程id 0是调度进程,常常被称为交换进程,该进程并不执行任何磁盘上的程序-是内核的一部分,也被称为系统进程。
进程id 1通常是init进程,在自举过程结束时由内核调用。init进程绝不会终止,它是一个普通的用户进程,但它以超级用户特权运行。
某些UNIX的虚存实现中,进程id 2是页精灵进程,也是内核进程。
2、fork函数
#include <sys/types.h> #include <unistd.h> pid_t fork(void);
由fork创建的新进程被称为子进程,该函数被调用一次,但返回两次。两次返回的区别是:子进程返回的是0,而父进程返回的是新子进程的进程ID。
子进程和父进程继续执行fork之后的指令。子进程是父进程的复制品。
#include "ourhdr.h" #include <sys/types.h> int glob = 6; char buf[] = "a write to stdout "; int main(void) { int var; pid_t pid; var=88; if (write(STDOUT_FILENO,buf,sizeof(buf)-1) != sizeof(buf)-1) err_sys("write error"); if ((pid=fork())<0) err_sys("fork error"); else if (pid == 0) { glob++; var++; }else sleep(2); printf("pid = %d, glob = %d, var = %d ",getpid(),glob, var); exit(0); }
编译执行
gcc fork.c
./a.out a write to stdout pid = 20346, glob = 7, var = 89 pid = 20345, glob = 6, var = 88
3、vfork函数
vfork用于创建一个新进程,而该新进程的目的是exec一个新程序。
vfork保证子进程先运行,在它调用exec或exit之后父进程才可能被调度运行。
#include "ourhdr.h" #include <sys/types.h> int glob = 6; /* external variable in initialized data */ int main(void) { int var; /* automatic variable on the stack */ pid_t pid; var = 88; printf("before vfork "); /* we don't flush stdio */ if ((pid = vfork()) < 0) { err_sys("vfork error"); } else if (pid == 0) { /* child */ glob++; /* modify parent's variables */ var++; _exit(0); /* child terminates */ } /* * Parent continues here. */ printf("pid = %d, glob = %d, var = %d ", getpid(), glob, var); exit(0); }
编译执行
./a.out before vfork pid = 20819, glob = 7, var = 89
子进程对变量glob和var做增1操作,结果改变了父进程的变量值,因为子进程在父进程的地址空间中运行。
另外如果子进程调用了exit而不是_exit,则刷新关闭了所有标准I/O流,包括标准输出,当父进程调用printf时,标准输出已经被关闭了,于是printf返回-1.
4、exit函数
在说明f o r k函数时,一定是一个父进程生成一个子进程。上面又说明了子进程将其终止状态返回给父进程。但是如果父进程在子进程之前终止,则将如何呢 ?
其回答是对于其父进程已经终止的所有进程,它们的父进程都改变为 i n i t进程。我们称这些进程由 i n i t 进程领养。
其操作过程大致是:在一个进程终止时,内核逐个检查所有活动进程,以判断它是否是正要终止的进程的子进程,如果是,则该进程的父进程 I D就更改为1 ( i n i t进程的I D )。
这种处理方法保证了每个进程有一个父进程。