fork小续
pid_t pid = fork();
1.根据fork的返回值区分父子进程:
fork 函数返回两次, >0 表示父进程,返回值为子进程ID; =0 表示子进程; <0 出错。
可用代码表示为:
if( ( pid = fork() ) < ){
//error
}
else if( pid == ){
//child
}
else{
//parent
}
2.进程调用wait等待子进程的退出:
系统调用wait做两件事:a.wait暂停调用它的进程直到子进程结束;b.wait取得子进程结束时传给exit的值。
系统调用wait的两个特征:a.wait阻塞调用它的进程直到子进程结束;b.wait返回结束进程的PID。
父进程必须要回收子进程,这时必须用wait/waitpid。不然当子进程结束后,它会变成僵死进程。用"ps -ef"命令查看,可看到其状态为<defunct>。所以,在fork()之后,我们一般执行:waitpid(pid,NUL,0);
参数1: pid表示到等待的子进程ID
参数2: NULL表示对子进程的结束状态不在意。
参数3: 控制选项
if( ( pid = fork() ) < ){
//error
}
else if( pid == ){
//child
...
}
else{
waitpid( pid, NULL, ); //用于等待子进程结束.
}
3. 问题提出——父进程被阻塞:
在调用waitpid后,父进程被阻塞,将不能执行其他的任务,这在监听程序中是不允许的。所以,必须要回收掉子进程,并且父进程不能被阻塞,要解决这个问题,我们可以fork两次,让子进程被init进程托管。
if( ( pid = fork() ) < ){
//error
}
else if( pid == ){
//first child exit !
if( fork() > ){
exit();
}
else{
//second child...
}
}
else{
waitpid( pid, NULL, );
//因为第一个子进程马上退出,所以waitpid在等到其结束时马上返回,第二
//个子进程将被init进程托管!
}
4.问题提升——子进程结束标志:
我们让子进程被init进程托管,也就等于对它放弃了控制权,没办法知道它什么时候结束的,以及结束的状态是什么。所以,为了父进程不被阻塞,并且能够及时的回收掉子进程。这时,用信号是个不错的选择。
int chld_count = ; void sigchd_exit(int signo)
{
if( waitpid(-, NULL, WNOHANG) > ) chld_count--; printf( "child count: %d\n", chld_count );
} void set_sigchd()
{
signal(SIGCHLD, sigchd_exit); //子程序退出的信号
signal(SIGINT, sigchd_exit); //Ctrl+C
signal(SIGTERM, sigchd_exit);
} //.....
set_sigchd()
if( ( pid = fork() ) < ){
//error
}
else if( pid == ){
//child
...
}
else{
chld_count++;
}
这时,在程序开始时设置信号处理函数,当子进程结束时,信号处理函数回收子进程,减少子进程计数。fork之后,父进程增加子进程计数。这样子,程序就实时的知道了子进程的个数。也就可以对止之后的fork做出一定的限制。
fork小续的更多相关文章
- 解决 adb.exe 停止工作小续
继adb 停止工作的问题之后,又碰见了adb 停止工作的问题. 在使用adb install app.apk 之后给出错误信息如下: * daemon not running. starting it ...
- Bitbucket Pull Request和fork
本文参考了http://blog.jobbole.com/76854/ Pull Request在Forking工作流中使用,这个也同样适用于小团队的开发协作和第三方开发者向开源项目的贡献.当你要 ...
- 小程序解决方案 Westore - 组件、纯组件、插件开发
数据流转 先上一张图看清 Westore 怎么解决小程序数据难以管理和维护的问题: 非纯组件的话,可以直接省去 triggerEvent 的过程,直接修改 store.data 并且 update,形 ...
- 1.1 Linux中的进程 --fork、孤儿进程、僵尸进程、文件共享分析
操作系统经典的三态如下: 1.就绪态 2.等待(阻塞) 3.运行态 其转换状态如下图所示: 操作系统内核中会维护多个队列,将不同状态的进程加入到不同的队列中,其中撤销是进程运行结束后,由内核收回. 以 ...
- 三周学会小程序第四讲:Heroku 绑定 Github 自动部署
这一讲是根据读者的反馈补充的一个讲解,好多读者反应安装 Heroku-cli 遇到问题,或者是操作繁琐,其实上一讲中提到的 Heroku 只是为了免费部署,而安装 Heroku-CLI只是为了部署,所 ...
- Git工作流总结
引用自:https://github.com/xirong/my-git/blob/master/git-workflow-tutorial.md 说明: 个人在学习Git工作流的过程中,从原有的 S ...
- Git工作流指南:Pull Request工作流
参考地址:http://blog.jobbole.com/76854/ Pull Requests是Bitbucket上方便开发者之间协作的功能.提供了一个用户友好的Web界面,在集成提交的变更到正式 ...
- [转]深入理解学习GIT工作流
深入理解学习Git工作流 字数13437 阅读2761 评论3 喜欢70 个人在学习git工作流的过程中,从原有的 SVN 模式很难完全理解git的协作模式,直到有一天我看到了下面的文章,好多遗留在心 ...
- printf 缓冲区问题
突然发现printf的问题,看了这个很有意思,学习一下 转自:http://blog.csdn.net/shanshanpt/article/details/7385649 昨天在做Linux实验的时 ...
随机推荐
- 2014ACM/ICPC亚洲区域赛牡丹江站现场赛-A ( ZOJ 3819 ) Average Score
Average Score Time Limit: 2 Seconds Memory Limit: 65536 KB Bob is a freshman in Marjar Universi ...
- linux 修改时间
实例:设置时间伟2008年8月8号12:00# date -s "2008-08-08 12:00:00"修改完后,记得执行clock -w,把系统时间写入CMOS date -s ...
- Centos 7 安装 Mysql5.7(压缩包方式)
今天装的了mysql,遇到了很多问题,好在最后一一解决了,现在记录在此,防止日后老路重走... 1.下载 当然是去官网,下一个linux下的版本,64位的 tar.gz,好吧这里贴个名字——[mysq ...
- 一个文件查看你选择 Run as Android applications 都干了啥
<?xml version="1.0" encoding="UTF-8"?> <project name="PushFastDemo ...
- 转 spring官方文档中文版
转 http://blog.csdn.net/tangtong1/article/details/51326887另附码云地址 https://gitee.com/free/spring-framew ...
- FPGA开发要懂得使用硬件分析仪调试——ILA
0. ILA概述在FPGA开发中,当我们写完代码,进行仿真,确定设计没有问题后,下载到硬件上一般都能按照我们的设计意愿执行相应功能.但这也并非绝对的,有时候你会遇到一些突然情况,比如时序问题或者仿真时 ...
- Python归并排序(递归实现)
为什么归并排序如此有用?1. 快捷和稳定归并排序成为⼀一个非常棒的排序算法主要是因为它的快捷和稳定.它的复杂度即使在最差情况下都是O(n log n).而快速排序在最差情况下的复杂度是O(n^2),当 ...
- 美团HD(7)-添加取消搜索按钮
DJSelectCityViewController.m #pragma mark - UISearchBar 代理方法 /** SearchBar开始编辑 */ - (void)searchBarT ...
- UML类图详解_泛化关系
泛化其实就是继承关系,还是比较简单的,那么我们就把之前有些问题的博客UML类图重新来实现一次. 依旧是这个图 下面我们来看一个例子 Account.h #include <cstdlib> ...
- 内核并发管理---spin lock
自旋锁最初是为了在smp系统上使用而设计. 1.在单处理器非抢占模式下,自旋锁不做任何事情. #ifdef CONFIG_PREEMPT_COUNT //支持抢占模式 #define pree ...