目录
一、fork函数
函数原型:pid_t fork(void);(pid_t 实际是int类型)
fork函数创建子进程,子进程的地址空间与父进程共享(不包括进程的栈区),但内核会将共享的空间设置为只读。
fork的返回值:
fork函数被父进程调用一次,但是却返回两次;一次是返回到父进程,一次是返回到新创建的子进程。(原来的进程返回新进程的pid,新进程中返回0)
写时拷贝:当任意一个进程试图修改共享空间中的数据,操作系统就会将需要修改的数据所在的页直接拷一份出来。
内核只为新生成的子进程创建虚拟空间结构,它们复制于父进程的虚拟空间结构,但是不为这些段分配物理内存,它们共享父进程的物理空间,当父子进程中有更改相应的段的行为发生时,再为子进程相应的段分配物理空间。
相同但是独立的地址空间
因为父进程和子进程是独立的进程,他们都有自己私有的地址空间,当父进程或者子进程单独改变时,不会影响到彼此,类似于c++的写实拷贝的形式自建一个副本。
fork的常规用法
1.一个父进程希望复制自己,使得子进程同时执行不同的代码段,例如:父进程等待客户端请求,生成一个子进程来等待请求处理。
2.一个进程要执行一个不同的程序。
fokr调用失败的原因
1.系统中有太多进程
2.实际用户的进程数超过限制
文件描述符:
1、fork之前打开的文件,子进程依旧可以通过其文件描述符访问文件;
2、父子进程操作fork之前打开的文件描述符会相互影响(读写偏移量);
3、fork之后,父进程关闭文件之后,子进程也可以通过文件描述符访问文件。(父子进程相互独立,单步执行)
fork之后,子进程会拷贝父进程的PCB结构,然后对PCB里面数据做修改。父进程的页表直接拷贝给子进程。父子进程共享所有的数据空间。
当父子进程任意一个进程试图修改数据时,操作系统就会将要修改的数据所在的页直接复制出来。
僵死进程:父进程未结束,子进程结束,并且父进程没有获取子进程的退出状态。这种进程,进程主体空间已经释放,只有PCB还没有释放。
解决方法:
1、父进程中调用wait或者waitpid获取子进程的退出状态,但会导致父进程在wait或waitpid调用时阻塞运行,芝罘岛子进程退出。
2、父进程调用signal(SIGCHLD,SIG_IGN),来忽略SIGCHLD信号,这样子进程结束后会由内核释放资源。
3、对子进程的退出捕获他们的退出信号SIGCHLD,父进程退出信号时,在信号处理函数中调用wait 或者waitpid操作来释放他们的资源。
阻塞运行:所需条件未准备好,函数不会返回,知道条件满足,从而造成进程执行阻塞。
非阻塞运行:函数调用,无论条件是否满足,函数都会返回。
孤儿进程:父进程结束,子进程没有结束。孤儿进程会被init收养,并为他们完成状态收集工作。
守护进程(精灵进程):
定义:常常在系统启动时自启,仅在系统关闭时才终止,生存期较长,一般在后台运行。
在Linux中通过ps -axj命令查看常用系统守护进程,其中最常见的init进程,负责各运行层次间的系统服务。
守护进程编程规则:
1、首先调用umask(mode_t umask())函数将文件模式创建屏蔽字为0;
2、调用fork()然后使父进程退出(exit);
3、调用setsid()创建一个新会话;
4、将当前目录改为根目录;
5、关闭不再需要的文件描述符;
6、某些守护进程打开/dev/null 使其具有文件描述符0,1,2,这样任何一个进程就不会产生其他不好的效果。
二、vfork函数
函数原型:pid_t vfork(void);
特点:父子进程共享数据段,并且保证子进程先于父子进程运行,在他调用exec或者_exit时,父进程才会被运行;
三、fork与Vfork的区别
1.、fork要拷贝父进程的数据段;而vfork则不需要完全拷贝父进程的数据段,在子进程没有调用exec和exit之前,子进程与父进程共享数据段
2、fork不对父子进程的执行次序进行任何限制;而在vfork调用中,子进程先运行,父进程挂起,直到子进程调用了exec或exit之后,父子进程的执行次序才不再有限制
四、结束子进程
结束子进程不用exit(0),而使用_exit(0)。这是因为_exit(0)在结束进程时,不对标准I/O流进行任何操作。而exit(0)则会关闭进程的所有标准I/O流。