fork可以在linux中创建子进程。先看man手册里面的东西:

SYNOPSIS
       #include <unistd.h>

pid_t fork(void);

DESCRIPTION
       fork()  creates  a new process by duplicating the calling process.  The
       new process, referred to as the child, is an  exact  duplicate  of  the
       calling  process,  referred  to as the parent    ......
RETURN VALUE
       On success, the PID of the child process is returned in the parent, and
       0  is returned in the child.  On failure, -1 is returned in the parent,
       no child process is created, and errno is set appropriately.

函数的作用的创建一个进程,这个函数会返回两次,可能有三种不同的返回值。 
1. 出错返回-1 
2. 返回0,表示是子进程 
3. 返回大于0,表示是父进程

下面的一个简单的实例:

#include<stdio.h>
#include<unistd.h>
#include<signal.h>
#include<sys/types.h>
#include<sys/wait.h> #define CHILDCOUNT 1 int main()
{
pid_t fpid[CHILDCOUNT ];
int i = ;
for(i = ; i < CHILDCOUNT; i++)
{
fpid[i] = fork();
if(fpid[i] < )
{
perror("fork");
return -;
}
else if(fpid[i] == )
{
printf("This is child process, id:%d, my father:%d\n", getpid(), getppid());
getchar();
return ;
}
else
{
printf("This is parent process, id:%d\n", getpid());
}
} getchar();
printf("main() ------\n");
return ;
}

补充一点知识

  • fork出来的子进程复制了父进程的内存空间(处理代码区都复制了),和父进程共享代码区
  • fork之后,父子进程谁先执行不确定
  • 子进程中新定义的变量和父进程没有任何关系
  • 子进程也会复制父进程文件描述符,但是不会复制文件表。而是共用一个offset
  • 如果父进程先挂了,子进程就变成孤儿进程了,它爹就会变成1号进程
  • 子进程挂了会给父进程发信号(SIGCHLD),如果父进程收到该信号没有及时处理。子进程就变成僵尸进程了,直到父进程处理了该信号或父进程也退出了。

如何避免僵尸进程的出现?
如之前所说,父进程应该及时处理子进程发出来的信号,并且去获取子进程的退出码。 
在父进程中使用wait或waitpid参数等待子进程退出。 
也可以在收到子进程退出的信号时用wait或waitpid等待。

#include<stdio.h>
#include<unistd.h>
#include<signal.h>
#include<sys/types.h>
#include<sys/wait.h> void sig_handle(int sig)
{
int ret = ;
printf("%s() +++\n", __func__);
#if 0
wait(NULL);
#else
while(ret = waitpid(-, NULL, WNOHANG))
{
if(ret < )
{
printf("waitpid failed\n");
}
else
{
printf("waitpid success\n");
break;
}
}
#endif
} int main()
{
signal(SIGCHLD, sig_handle); // 标记A
pid_t fpid[];
int i = ;
for(i = ; i < ; i++)
{
fpid[i] = fork();
if(fpid[i] < )
{
perror("fork");
return -;
}
else if(fpid[i] == )
{
printf("This is child process, id:%d, my father:%d\n", getpid(), getppid());
return ;
}
else
{
printf("This is parent process, id:%d\n", getpid());
}
} getchar();
printf("main() ------\n");
return ;
}

假如没有signal(SIGCHLD, sig_handle);这个语句,函数运行时,5个子进程马上就结束了(父进程不能退出了)。这时候去查看可以看到有5个僵尸进程。 
如下:

xcy@xcy-virtual-machine:~/test/sock4$ ps -ef | grep test
xcy : pts/ :: ./test
xcy : pts/ :: [test] <defunct>
xcy : pts/ :: [test] <defunct>
xcy : pts/ :: [test] <defunct>
xcy : pts/ :: [test] <defunct>
xcy : pts/ :: [test] <defunct>
xcy : pts/ :: grep --color=auto test
xcy@xcy-virtual-machine:~/test/sock4$

加上那句话就没有了

xcy@xcy-virtual-machine:~/test/sock4$ ps -ef | grep test
xcy : pts/ :: ./test
xcy : pts/ :: grep --color=auto test
xcy@xcy-virtual-machine:~/test/sock4$

再来一些有意思的东西
我们知道fork会出现两个进程,两个进程都会往下执行,看下面的代码:

#include<stdio.h>
#include<unistd.h>
int main()
{
fork();
fork();
fork();
printf("+\n");
}

这样就相当于最开始主进程创建两个进程A和B。 
A又创建两个进程A1,A2。B又创建两个进程B1,B2。 
最后,A1,A2,B1,B2都创建两个进程,所以最后会有8个进程,+会打印8次。 
这样就想细胞的二次分裂一样,3次分裂周期之后就有了8个细胞。

再来看个更难的:

#include<stdio.h>
#include<unistd.h> void fun()
{
fork();
// fork() || fork() && fork(); // 第一个fork返回大于0,后面两个就不执行了
fork() && fork() || fork();
fork();
printf("+\n");
} int main()
{
fun2();
}

这样一共能创建几个进程呢,+会打印几次呢。我们先一步一步来分析,假定所有的fork都会成功: 
第一次分裂成A和B。A进程创建的进程和B创建的进程数目肯定是一样的。

就先像下面这样分析:

fork() && fork() || fork(); // line 1
fork(); // line 2

要注意&&运算符和||运算符: 
对于&&来说:第一个表达式如果为0,后面的表达式就不用算了 
对于||来说:第一个表达式如果为1,后面的表达式也不用算了。 
为了更好分析,上面的fork一次标号为fork1,fork2,fork3,fork4。 
先看line1。fork1会返回一个非0,和一个0.

  • 1.返回0,则直接进入line2,fork4创建两个进程
  • 2.返回大于0,执行fork2。
  • 2.1 fork2返回0,则需要执行fork3,再接着执行fork4.这里会创建4个进程
  • 2.2 fork2返回大于0,也会执行fork3和fork4。相当于也创建4个进程。

综上,第一次的A进程会产生10个进程,同理,B也会有10个进程。所以一共会有20个进程。

关于signal和fork的思考的更多相关文章

  1. 12.详解Condition的await和signal等待通知机制

    1.Condition简介 任何一个java对象都天然继承于Object类,在线程间实现通信的往往会应用到Object的几个方法,比如wait(),wait(long timeout),wait(lo ...

  2. linux signal函数遇到的问题

    1.关于signal函数的定义 signal最开始的原型是这: void (*signal(int signo, void (*func)(int)))(int);看过下面两行,了解到上面这一行是这个 ...

  3. 详解Condition的await和signal等待/通知机制

    本人免费整理了Java高级资料,涵盖了Java.Redis.MongoDB.MySQL.Zookeeper.Spring Cloud.Dubbo高并发分布式等教程,一共30G,需要自己领取.传送门:h ...

  4. Linux Pthread 深入解析(转-度娘818)

    Linux Pthread 深入解析   Outline - 1.线程特点 - 2.pthread创建 - 3.pthread终止         - 4.mutex互斥量使用框架         - ...

  5. Network client/server

    <Beginning Linux Programming_4th>  chapter 15 Sockets 1  A simple local client/server 1)  clie ...

  6. 关于synchronized和lock 的使用及其在线程间的通信

    题目要求:子线程循环10次,接着主线程循环100次,接着又回到子线程循环10次,接着再回到主线程又循环100,如此循环50次 synchronized的使用 import java.util.conc ...

  7. GDB常用命令使用说明(一)

    本文由霸气的菠萝原创,转载请注明出处:http://www.cnblogs.com/xsln/p/gdb_instructions1.html 全部关于gdb的文章索引请点这里 GDB(GNU Deb ...

  8. 玩转Nodejs的集群

    在Nodejs中使用集群还是不容易的.Javascript的单线程属性让nodejs下的应用很难使用现代机器的多核特性.比如下面的代码实现了一个http服务器的主干部分.这部分代码只会执行在一个线程上 ...

  9. 浅谈NodeJS多进程服务架构基本原理

    阅读目录 一:nodejs进程进化及多进程架构原理 二:node中child_process模块实现多进程 三:父子进程间如何通信? 四:理解cluster集群 回到顶部 一:nodejs进程进化及多 ...

随机推荐

  1. python爬虫(一)_爬虫原理和数据抓取

    本篇将开始介绍Python原理,更多内容请参考:Python学习指南 为什么要做爬虫 著名的革命家.思想家.政治家.战略家.社会改革的主要领导人物马云曾经在2015年提到由IT转到DT,何谓DT,DT ...

  2. Python学习笔记整理总结【语言基础篇】

    一.变量赋值及命名规则① 声明一个变量及赋值 #!/usr/bin/env python # -*- coding:utf-8 -*- # _author_soloLi name1="sol ...

  3. Vue.js优雅的实现列表清单

        一.Vue.js简要说明 Vue.js (读音 /vjuː/) 是一套构建用户界面的渐进式框架.与前端框架Angular一样, Vue.js在设计上采用MVVM模式,当View视图层发生变化时 ...

  4. PowerShell安全修改Windows 10 登陆背景图

    PowerShell安全修改Windows 10 登陆背景图 可以把登陆的背景图换掉,主要是修改操作pri文件 $priPath = "$env:windir\SystemResources ...

  5. javascript第四章--面向对象的程序设计

    ① 理解对象 ② 创建对象 ③ 继承

  6. 微信小程序开发《三》:微信小程序请求不能使用session的原因及解决办法

    本人在前面的微信小程序开发<二>中提到要想在服务端保持状态需要在客户端第一次请求服务器的时候给客户端返回一个sessionid,由客户端在本地保存,下次请求的时候在header里面带上这个 ...

  7. 对java泛型的理解

    正确的应用java泛型的特性可以更好的实现编程的开闭原则(对扩展开放,对修改关闭),这得益于java泛型提供的在程序运行时获取对象声明类型的特性. 静态语言的特性是在程序编译前进行声明,这样程序在编译 ...

  8. Redis+Tomcat+Nginx集群实现Session共享,Tomcat Session共享

    Redis+Tomcat+Nginx集群实现Session共享,Tomcat Session共享 ============================= 蕃薯耀 2017年11月27日 http: ...

  9. iOS面试题最全梳理

    OC的理解与特性 OC作为一门面向对象的语言,自然具有面向对象的语言特性:封装.继承.多态.它既具有静态语言的特性(如C++),又有动态语言的效率(动态绑定.动态加载等).总体来讲,OC确实是一门不错 ...

  10. spark2的编译

    0.操作系统 centos:6.4 hadoop:2.5.0-cdh5.3.6 1.为什么要编译 spark 源码? 学习spark的第一步 就应该是编译源码,后期修改和调试,扩展集成的功能模块 2. ...