1. fork函数

  fork函数用于克隆一份当前的进程资源,调用fork函数之后,进程一分为二,并且两个进程的资源是一样的(只是资源内容完全一样,并不是同一份资源)。fork函数的函数原型为:pid_t fork(void); 需要包含unistd.h,返回值pid_t类型实际上就是int型

  在调用fork函数之后,进程被一分为二,他们的资源都是相同的,如果在调用fork前,程序打开了某个文件,那么fork之后会出现两个一样的文件描述符(并非出现两个相同的文件),如果在调用fork前程序缓冲区中保存了某些数据,那么fork之后就会出现两个相同的缓冲区,里面存放的数据都是一模一样的。在fork之后他们可以进行不同的操作,他们可以去修改同一个文件,也可以改变自己的缓冲区。这个机制就好像克隆人一样,从本体克隆出来一个克克隆体,在克隆体被克隆出来那一刻,克隆人与本体是一模一样的,克隆人与本体有着相同的记忆(虽然他并没有真正经历过这些事情)。在克隆人被克隆出来之后,他们经历的事情就会不一样了,比如本体去了A地,他记住了A第发生的事情,而克隆体去了B地,他便记住了B地发生的事情,就好像fork函数执行完的那一刻,缓冲区的数据是一样的,而之后程序的走向可以让缓冲区的数据不一样。如果本体在以前有一个女朋友,那她同样也是克隆体的女朋友,但是“女朋友”这个人只有一个,这就好像fork之前打开了某个文件,在fork之后文件描述符变成了两份,但文件其实只有一份是相同的道理。

  实际上在fork函数返回之后,两个程序已经不一样了,区别就是fork的返回值。返回值大于0的是父进程,其返回值是子进程的ID,返回值等于0的是子进程(但这并不是说子进程的ID是0),返回值小于0表示fork失败。通过geipid函数可以得到自己的进程ID(PID),通过getppid函数可以得到父进程的ID,这两个函数是不会执行失败的,因为在函数的描述里写道:“These functions are always successful.”,这两个函数的原型如下:

pid_t getpid(void);
pid_t getppid(void);

  通过一个简单的例子说明:

 1 #include <stdio.h>
2 #include <sys/types.h>
3 #include <unistd.h>
4
5 int main(int argc, const char *argv[])
6 {
7 int pid = 0; /* fork函数的返回值 */
8 int mypid = 0; /* PID */
9 int myppid = 0; /* PPID */
10
11 pid = fork();
12
13 if (pid > 0) { /* 父进程 */
14 sleep(1);
15 printf("##################\n");
16 printf("I'm parent\n");
17 printf("pid = %d\n", pid);
18 printf("mypid = %d\n", getpid());
19 printf("myppid = %d\n\n", getppid());
20 } else if (pid == 0) { /* 子进程 */
21 printf("##################\n");
22 printf("I'm child\n");
23 printf("pid = %d\n", pid);
24 printf("mypid = %d\n", getpid());
25 printf("myppid = %d\n", getppid());
26 }
27 }

执行结果如下图:

2. 进程ID

  由上图可知,子进程打印pid的值为0,而他自己的PID为4244,他的父进程ID为4243,父进程打印的pid值为4244,这恰好是子进程通过getpid函数得到的PID值,而父进程的PID为4243,父进程的父进程PID为2732。

  上边的代码在父进程处加一个sleep(1)的目的是让子进程先运行,之后子进程退出,父进程再运行,然后父进程退出,如果父进程退出了,子进程再获取父进程的的ID会发生什么呢,代码作如下改动:

 1 #include <stdio.h>
2 #include <sys/types.h>
3 #include <unistd.h>
4
5 int main(int argc, const char *argv[])
6 {
7 int pid = 0; /* fork函数的返回值 */
8 int mypid = 0; /* PID */
9 int myppid = 0; /* PPID */
10
11 pid = fork();
12
13 if (pid > 0) { /* 父进程 */
14 sleep(1);
15 printf("##################\n");
16 printf("I'm parent\n");
17 printf("pid = %d\n", pid);
18 printf("mypid = %d\n", getpid());
19 printf("myppid = %d\n\n", getppid());
20 } else if (pid == 0) { /* 子进程 */
21 printf("##################\n");
22 printf("I'm child\n");
23 printf("pid = %d\n", pid);
24 printf("mypid = %d\n", getpid());
25 printf("myppid = %d\n", getppid());
26 sleep(2);
27 printf("I'm child\n");
28 printf("myppid = %d\n", getppid());
29 }
30 }

  运行结果如下图:

  从运行结果发现,如果父进程退出而子进程还没退出,那么子进程的父进程PID将变为1,子进程此时变为孤儿进程,PID为1的进程就是linux系统下负责收留孤儿进程的进程。

  通过前面的例子可以看出,fork函数虽然将资源一分为二了,但分出来的两部分还是不能当成完全相同的两部分来看待,他们之间存在父子关系。如果父进程退出,那么子进程会变成孤儿进程,如果子进程退出,那么子进程会给父进程发一个信号,这在下一章讨论。因此当使用fork的时候要考虑进程是否会退出,从而选择合适的分支。

2、fork函数与进程ID的更多相关文章

  1. 关于fork( )函数父子进程返回值的问题

    fork()是linux的系统调用函数sys_fork()的提供给用户的接口函数,fork()函数会实现对中断int 0x80的调用过程并把调用结果返回给用户程序. fork()的函数定义是在init ...

  2. fork()函数、进程表示符、进程位置

    linux.centos6.5 fork()函数:作用于创建子进程.返回值有两个,一个是向父进程返回它的pid,一个是返回0: eg1: #include<stdio.h> #includ ...

  3. linux多进/线程编程(2)—— fork函数和进程间“共享”数据

    参考: 1.博客1:https://www.pianshen.com/article/4305691855/ fork:在原进程的基础上"分叉"出一个子进程,即创建一个子进程. N ...

  4. 通过fork函数创建进程的跟踪,分析linux内核进程的创建

    作者:吴乐 山东师范大学 <Linux内核分析>MOOC课程http://mooc.study.163.com/course/USTC-1000029000 一.实验过程 1.打开gdb, ...

  5. Linux编程之fork函数

    在Linux中,fork函数的功能就是在一个进程中创建一个新的进程,当前调用fork函数的进程就是产生的新进程的父进程,新进程在以下也称为子进程.在新进程生成之后就会在系统中开始执行. 函数原型:pi ...

  6. [转帖]Linux下fork函数及pthread函数的总结

    Linux下fork函数及pthread函数的总结 https://blog.csdn.net/wangdd_199326/article/details/76180514 fork Linux多进程 ...

  7. 深入浅出--UNIX多进程编程之fork()函数

    0前言 上周都在看都在学习unix环境高级编程的第八章--进程控制.也就是这一章中.让我理解了unix中一些进程的原理.以下我就主要依照进程中最重要的三个函数来进行解说.让大家通过阅读这一篇文章彻底明 ...

  8. C/C++网络编程7——多进程服务器端之fork函数

    通过前面几节的内容,我们已经可以实现基本的C/S结构的程序了,但是当多个客户端同时向服务器端请求服务时,服务器端只能按顺序一个一个的服务,这种情况下,客户端的用户是无法忍受的.所以虚实现并发的服务器端 ...

  9. 进程控制之fork函数

    一个现有进程可以调用fork函数创建一个新进程. #include <unistd.h> pid_t fork( void ); 返回值:子进程中返回0,父进程中返回子进程ID,出错返回- ...

随机推荐

  1. 使用pdf2htmlEX将pdf文件转为html

    https://github.com/coolwanglu/pdf2htmlEX 参考github文档,转换出来的的效果貌似很好,可以参考OFFICE 文档转换为html在线预览. pdf2swf 和 ...

  2. easyui中加载table列表数据 第一次有数据第二次没有数据问题

    $('#allUsingProductTable').datagrid({  加载数据时,第二加载时table会发生变化会出现找不到问题.如果是弹框没有影响,弹框出现出现列表每次都会执行销毁方法. 解 ...

  3. Flowable—多实例任务:会签

    多实例任务 会签 什么是会签? 举个例子:比如我们有一个任务 可能需要多人审批,审批通过的条件可能比较多样,比如五个人审批.3个人审批过了就算过了,再或者有一个人权限比较高,拥有一票否决权. 即是其余 ...

  4. java动态代理实现与原理详细分析(转)

    关于Java中的动态代理,我们首先需要了解的是一种常用的设计模式--代理模式,而对于代理,根据创建代理类的时间点,又可以分为静态代理和动态代理. 一.代理模式    代理模式是常用的java设计模式, ...

  5. [leetcode]114. Flatten Binary Tree to Linked List由二叉树构建链表

    /* 先序遍历构建链表,重新构建树 */ LinkedList<Integer> list = new LinkedList<>(); public void flatten( ...

  6. 使用Arduino点亮ESP-01S,ESP8266-01S上的板载LED

    因为在开发ESP-01s远程控制中觉得接线麻烦,又因为ESP-01s板子上带有LED灯,那就先点亮板载LED,  如图所示: 打开Arduino 把代码copy进去,再编译烧录,就可以看见LED灯每隔 ...

  7. Cookie和登录注册

    1. 什么是Cookie? 服务器通过 Set-Cookie 头给客户端一串字符串 客户端每次访问相同域名的网页时,必须带上这段字符串 客户端要在一段时间内保存这个Cookie Cookie 默认在用 ...

  8. 10. C++对象模型和 this 指针

    1. 成员变量和成员函数分开存储 在C++中,类内的成员变量和成员函数分开存储,只有非静态成员变量才属于类的对象上 空对象占用内存空间为:1 ----> C++编译器会给每个空对象也分配一个字节 ...

  9. 一、linux安装mysql

    一.下载mysql免编译包: wget http://cdn.mysql.com/archives/mysql-5.6/mysql-5.6.33-linux-glibc2.5-x86_64.tar.g ...

  10. Java并发编程实战(3)- 互斥锁

    我们在这篇文章中主要讨论如何使用互斥锁来解决并发编程中的原子性问题. 目录 概述 互斥锁模型 互斥锁简易模型 互斥锁改进模型 Java世界中的互斥锁 synchronized中的锁和锁对象 synch ...