Linux内核通过一个被称为进程描述符的task_struct结构体来管理进程,这个结构体包含了一个进程所需的所有信息,包括数据段,代码段,堆栈段的地址,当前进程的环境变量,文件的描述符等,在创建进程的时候,会为进程创建进程描述符。
1.fork系统调用
fork用来创建子进程,fork调用一次返回两次,有三种不同的返回值,在父进程中返回值为子进程的PID,子进程中返回值为0,如果出现错误fork返回一个负值,程序正是根据返回值的不同来判断当前是子进程还是父进程中,执行不同的代码。
函数调用过程:操作系统会先创建一个进程描述块,然后把父进程的所有进程描述符的信息精确拷贝进来,和父进程一样(PID除外),代码段共享,数据段和堆栈段复制(复制采用copy_on_write技术),所有的寄存器全部精确拷贝,文件描述符精确拷贝,一旦子进程开始运行,则新旧进程的地址空间已经分开,两者独立运行,但是fork父、子进程执行顺序不确定。
优点:子进程的执行独立于父进程,具有良好的并发性。缺点两者的通信需要有专门的通信机制,如pipe、fifo等。
2.vfork系统调用
vfork也是用来创建子进程,但是vfork创建的子进程完全运行在父进程的地址空间上,子进程对虚拟地址空间任何数据的修改都为父进程所见,没有独立空间,vfork子进程与父进程共享数据段。
vfork创建子进程后,父进程会被阻塞,子进程先运行,父进程后运行,它调用exec或exit之后父进程才可能被调度运行,如果在调用这两个函数之前子进程依赖于父进程的进一步动作,则会导致死锁。
优点:当创建子进程的目的仅仅是为了调用exec执行另一个程序时,子进程不会对父进程的地址空间有任何引用,因此,此时对地址空间的复制是多余的,通过vfork可以减少不必要的开销。
3.clone系统调用
clone创建进程较为复杂,clone可以有选择性的继承父进程的资源,可以向vfork一样和父进程共享一个虚存空间,从而使创造的是线程,也可以使创造出来的进程不是父子关系,而是兄弟关系。
4.exec函数族
exec用被执行的程序替换调用它的程序,与fork相比,fork创建一个新的进程,产生一个新的PID,exec启动一个新的程序替换当前的进程PID不变。
参考:
http://www.360doc.com/content/11/0502/11/6580811_113691501.shtml
http://blog.csdn.net/jianchi88/article/details/6985326
http://blog.csdn.net/ljianhui/article/details/10089345