fork
子进程和父进程共享很多资源,除了打开文件之外,很多父进程的其他性质也由子进程继承:
• 实际用户ID、实际组ID、有效用户ID、有效组ID。
• 添加组ID。
• 进程组ID。
• 对话期ID。
• 控制终端。
• 设置-用户- ID标志和设置-组- ID标志。
• 当前工作目录。
• 根目录。
• 文件方式创建屏蔽字。
• 信号屏蔽和排列。
• 对任一打开文件描述符的在执行时关闭标志。
• 环境。
• 连接的共享存储段。
• 资源限制。
父、子进程之间不继承的是:
• fork的返回值。
• 进程ID。
• 不同的父进程ID。
• 子进程的tms_utime , tms_stime ,tms_cutime以及tms_ustime设置为0。
• 父进程设置的锁,子进程不继承。
• 子进程的未处理的闹钟(alarm)被清除。
• 子进程的未处理的信号集设置为空集。
父、子进程共享的是:(既子进程改变父进程也改变)
fork之前打开的文件描述符的偏移标记。
父、子进程不共享的是:(既子进程改变父进程不变)
除以上都不共享,子进程只是复制了父进程的数据。
fork失败的两个主要原因是:( a )系统中已经有了太多的进程(通常意味着某个方面出了问题),或者( b )该实际用户I D的进程总数超过了系统限制。回忆表2 - 7,其中CHILD_MAX规定了每个实际用户ID在任一时刻可具有的最大进程数。
vfork
vfork(建立一个新的进程)
相关函数 wait,execve
头文件 #include<unistd.h>
定义函数 pid_t fork(void);
函数说明
vfork()会产生一个新的子进程.但是vfork创建的子进程与父进程共享数据段,而且由vfork创建的
子进程将先于父进程运行.fork()的使用详见百度词条fork().
vfork()用法与fork()相似.但是也有区别,具体区别归结为以下3点:
1. fork():子进程拷贝父进程的数据
段,代码段. vfork():子进程与父进程共享数据段.
2. fork():父子进程的执行次序不确定.
vfork():保证子进程先运行,在调用exec或exit之前与父进程数据是共享的,在它调用exec
或exit之后父进程才可能被调度运行。
3. vfork()保证子进程先运行,在她调用exec或exit之后父进程才可能被调度运行。如果在
调用这两个函数
之前子进程依赖于父进程的进一步动作,则会导致死锁。
返回值
如果vfork()成功则在父进程会返回新建立的子进程代码(PID),
而在新建立的子进程中则返回0。如果vfork失败则直接返回-1,失
败原因存于errno中。
错误代码
EAGAIN 内存不足
ENOMEM 内存不足,无法配置核心所需的数据结构空间。
例如:
#include <unistd.h>
#include <stdio.h>
#include <stdlib.h>
int glob=6;
int main(void)
{
int var;
pid_t pid;
var=88;
if((pid=vfork())<0) return 0;
if(pid==0)
{
glob++;
var++;
_exit(0);
}
printf("glob=%d,var=%d\n",glob,var);
exit(0);
}
结果:
glob=%d,var=%d