fork
fork()
调用一次返回两次:父进程返回子进程pid,子进程返回0。
进程是树形结构,子进程可随时通过getppid()获取父进程;而父进程只能在创建时获取子进程号。
返回值小于0,则表示创建失败。一般是由于用户ID的进程总数超过了系统限制。
子进程使用父进程 堆、栈、数据空间 等存储空间的副本。但代码正文段及文件共享,例如文件偏移量。
这里存储空间多采用写时复制的方式Copy-On-Write。一开始共享存储空间(此时有写保护,进程拥有只读权限),直到子进程试图修改某个区域时,进行复制。复制单位一般为页大小。
常见用法:
1、父进程希望子进程复制自己,后处理不同的代码段。如网络请求:父进程循环等待请求,有请求接入则将后续操作交给子进程;
2、父进程需要执行另一个不同的程序。一般交给子进程,fork()后exec()。如shell操作,python内的subprocess。
例子:
#include <unistd.h> #include <stdio.h> #include <sys/types.h> int main() { int money = 100; pid_t kid = fork(); if(kid < 0) { printf("wrong!"); }else if (kid == 0) { printf("I'm kid! pid=[%d]\n", getpid()); money += 10; }else { printf("I'm parent! pid=[%d] \n", getpid()); money -= 10; } printf("pid=[%d], ppid=[%d], money=[%d] \n", getpid(), getppid(), money); getchar(); return 0; }
运行结果:
I'm parent! pid=[3282] pid=[3282], ppid=[3139], money=[90] I'm kid! pid=[3283] pid=[3283], ppid=[3282], money=[110]