fork函数:

#include <unistd.h>

       pid_t fork(void);

fork用来创建一个子进程;

特点

fork调用后会返回两次,子进程返回0,父进程返回子进程的进程ID;fork返回后,子进程和父进程都从fork函数的下一条语句開始运行;

注意

fork之后,操作系统会复制一个与父进程全然同样的子进程,虽说是父子关系。可是在操作系统看来,他们更像兄弟关系,这两个进程共享代码空间,可是数据空间是互相独立的,子进程数据空间中的内容是父进程的完整拷贝。指令指针也全然同样。子进程拥有父进程当前执行到的位置(两进程的程序计数器pc值同样,也就是说,子进程是从fork返回处開始执行的),但有一点不同,假设fork成功,子进程中fork的返回值是0,父进程中fork的返回值是子进程的进程号,假设fork不成功,父进程会返回错误。

    能够这样想象,2个进程一直同一时候执行,并且步调一致,在fork之后,他们分别作不同的工作,也就是分岔了。

这也是fork为什么叫fork的原因。至于子进程和父进程哪个先执行。这是不确定的,取决于操作系统。假设用vfork,则能够保证子进程先执行
完毕后父进程在执行。

上面的注意中我们知道。子进程数据空间中的内容是父进程的完整拷贝。就是说子进程中对数据的操作是不会影响父进程的,以下的样例能够说明这一个特点:

#include <stdio.h>
#include <unistd.h> int main()
{
int i = 10;
pid_t pid;
printf("Father's pid:%d\n", getpid()); pid = fork();
if(pid < 0)
{
perror("fork failure!");
return -1;
}
else if(pid == 0)
{
while(1)
{
i++;
printf("Child's i = %d\n", i);
sleep(1);
}
}
else
{
printf("Child's pis:%d\n", pid);
while(1)
{
printf("Father's i = %d\n", i);
sleep(1);
}
sleep(1);
} return 0;
}

执行结果:

Father's pid:12148

Child's pis:12149

Father's i = 10

Child's i = 11

Father's i = 10

Child's i = 12

Father's i = 10

Child's i = 13

........

另一点要注意,假设父进程中打开了文件,即内核给应用程序返回一个文件描写叙述符,子进程和父进程的文件描写叙述符所相应的文件表项是共享的,这意味着子进程对文件的读写直接影响父进程的文件位移量(反之同理)。

进程中调用fd2 = dup(fd1) 产生的新的fd2所指向的文件表项和fd1指向的文件表项是同样的;

wait和waitpid函数:

wait和waitpid用来等待子进程结束;

假设没有子进程。则wait出错返回;

    有子进程。子进程正在执行,则堵塞。等待子进程结束;

    假设子进程已经结束,则得到结束的子进程的信息,并返回;

为什么要用wait和waitpid函数?

假设父进程先结束。子进程则成为孤儿进程,此时init进程(id为1)会成为子进程的新的父进程;

假设子进程先结束,则子进程会成为僵死进程!

僵死进程本身并不占有CPU资源。可是它占用了进程表项,假设有非常多僵死进程,那么非常多正常的进程就无法注冊进进程表了;因此。我们必需要对僵死进程进行回收,就用wait和waitpid;

   waitpid()会临时停止眼下进程的运行,直到有信号来到或子进程结束。

      假设在调用 wait()时子进程已经结束,则 wait()会马上返回子进程结束状态值。
      子进程的结束状态值会由參数 status 返回,而子进程的进程识别码也会一快返回。

      假设不在意结束状态值,则參数 status 能够设成 NULL。

   參数 pid 为欲等待的子进程识别码,其它数值意义例如以下:
      pid < -1 等待进程组识别码为 pid 绝对值的不论什么子进程。
      pid = -1 等待不论什么子进程,相当于 wait()。

      pid = 0  等待进程组识别码与眼下进程同样的不论什么子进程。

      pid > 0  等待不论什么子进程识别码为 pid 的子进程。


  參数 option 能够为 0 或以下的 OR 组合:
     WNOHANG 假设没有不论什么已经结束的子进程则立即返回, 不予以等待。
     WUNTRACED 假设子进程进入暂停运行情况则立即返回,但结束状态不予以理会。


  子进程的结束状态返回后存于 status,底下有几个宏可判别结束情况:
     WIFEXITED(status)假设子进程正常结束则为非 0 值。

     WEXITSTATUS(status)取得子进程 exit()返回的结束代码,通常会先用 WIFEXITED 来推断是否正常结束才干使用此宏。

     WIFSIGNALED(status)假设子进程是由于信号而结束则此宏值为真
     WTERMSIG(status) 取得子进程因信号而中止的信号代码,通常会先用 WIFSIGNALED 来推断后才使用此宏。

     WIFSTOPPED(status) 假设子进程处于暂停运行情况则此宏值为真。

一般仅仅有使用 WUNTRACED 时才会有此情况。

     WSTOPSIG(status) 取得引发子进程暂停的信号代码,通常会先用 WIFSTOPPED 来推断后才使用此宏。
     假设运行成功则返回子进程识别码(PID) ,假设有发生错误则返回返回值-1。失败原因存于 errno 中。

样例:

#include <stdio.h>
#include <unistd.h> int main()
{
int i = 10;
pid_t pid;
int status; printf("Father's pid:%d\n", getpid()); pid = fork();
if(pid < 0)
{
perror("fork failure!");
return -1;
}
else if(pid == 0)
{
i++;
printf("Child's i = %d\n", i);
sleep(7);
}
else
{
printf("Child's pis:%d\n", pid);
printf("Father's i = %d\n", i);
sleep(2);
// wait(&status);
} return 0;
}

上面的程序假设不使用wait函数对子进程进行回收,则父进程2秒正常结束后。子进程的父进程会变为init进程。能够用ps -l命令查看,达到执行的时间7秒后,子进程正常结束;假设使用了wait,则wait会使父进程等待子进程结束,子进程结束后一起退出,避免了僵死进程的产生。

Linux系统编程_8_进程控制之fork_wait_waitpid函数的更多相关文章

  1. Linux系统编程之进程控制(进程创建、终止、等待及替换)

    进程创建 在上一节讲解进程概念时,我们提到fork函数是从已经存在的进程中创建一个新进程.那么,系统是如何创建一个新进程的呢?这就需要我们更深入的剖析fork函数. 1.1 fork函数的返回值 调用 ...

  2. linux c编程:进程控制(三)_exec函数

    fork()函数通过系统调用创建一个与原来进程(父进程)几乎完全相同的进程(子进程是父进程的副本,它将获得父进程数据空间.堆.栈等资源的副本.注意,子进程持有的是上述存储空间的“副本”,这意味着父子进 ...

  3. Linux系统编程:进程控制

    一.进程相关操作与编程对应函数 1.进程创建:两种方式来实现. ①fork:创建一个子进程,父子进程共享一份代码程序,但是各有一份独立的数据,为了效率和保持数据的独立采用写时复制技术(COW).运行无 ...

  4. linux系统编程:进程控制(fork)

    在linux中,用fork来创建一个子进程,该函数有如下特点: 1)执行一次,返回2次,它在父进程中的返回值是子进程的 PID,在子进程中的返回值是 0.子进程想要获得父进程的 PID 需要调用 ge ...

  5. linux系统编程之进程(一)

    今天起,开始学习linux系统编程中的另一个新的知识点----进程,在学习进程之前,有很多关于进程的概念需要了解,但是,概念是很枯燥的,也是让人很容易迷糊的,所以,先抛开这些抽象的概念,以实际编码来熟 ...

  6. Linux系统编程——Daemon进程

    目录 Daemon进程介绍 前提知识 Daemon进程的编程规则 Daemon进程介绍 Daemon运行在后台也称作"后台服务进程". 它是没有控制终端与之相连的进程.它独立与控制 ...

  7. linux系统编程之进程(六):父进程查询子进程的退出,wait,waitpid

    本节目标: 僵进程 SIGCHLD wait waitpid 一,僵尸进程 当一个子进程先于父进程结束运行时,它与其父进程之间的关联还会保持到父进程也正常地结束运行,或者父进程调用了wait才告终止. ...

  8. linux系统编程之进程(五)

    今天继续学习系统编程,学习的主题还是进程,今天主要讨论的是守护进程相关的概念,开始进入正题: 什么是守护进程:       守护进程的创建步骤: 在描述它之前,首先得先了解两个概念:进程组.会话期: ...

  9. linux系统编程之进程(二):进程生命周期与PCB(进程控制块)

    本节目标: 进程状态变迁 进程控制块 进程创建 进程撤消 终止进程的五种方法 一,进程状态变迁 进程的三种基本状态 就绪(Ready)状态 当进程已分配到除CPU以外的所有必要的资源,只要获得处理机便 ...

随机推荐

  1. mysql 压力测试之批量插入自增字段不连续问题

    Gaps in auto-increment values for “bulk inserts” With innodb_autoinc_lock_mode set to 0 (“traditiona ...

  2. 【Codeforces Round #451 (Div. 2) A】Rounding

    [链接] 我是链接,点我呀:) [题意] 在这里输入题意 [题解] 模拟 [代码] /* 1.Shoud it use long long ? 2.Have you ever test several ...

  3. RFID的工作流程

    工作流程 1.阅读器通过发射天线发送一定频率的射频信号, 2.当射频卡进入发射天线工作区域时产生感应电流,射频卡获得能量被激活: 3.射频卡将自身编码等信息通过卡内置发送天线发送出去 4.系统接收天线 ...

  4. menu-普通menu弹出框样式

    今天接触到了menu弹出框样式.主要就是在theme下进行调整.现在把接触到的知识点总结一下. 在theme中,跟menu有关的几个属性如下 <item name="panelBack ...

  5. drawable-实现图片旋转

    今天因为需要,所以要让一个图片随着某种需要进行旋转.但是,又不能一张张的做动态图片.所以就在网上找了这么个方法.但是,这个方法有个问题,就是虽然能实现图片的旋转.但是,图片旋转以后会进行缩放.具体原因 ...

  6. AJAX有关的请求协议及HTTP报文

    URI:统一资源标识符 URI=URL+URNURL:统一资源定位符URN:统一资源名称 上边的图片编号对应下边的编号说明: 1.HTTP(占90%市场)/HTTPS/FTP 传输协议(可以理解为快递 ...

  7. django-rest-framework框架 第二篇 之Mixin扩展类

    Mixin扩展类     ['列表操作','过滤','搜索','排序'] <一>:<1>创建项目: 配置 urls 主路由    配置model文件(举个例子,就以book为模 ...

  8. HTTP--Request Headers及Cookies

    简介: HTTP客户程序(例如浏览器),向服务器发送请求的时候必须指明请求类型(一般是GET或者POST).如有必要,客户程序还可以选择发送其他的请求头.大多数请求头并不是必需的,但Content-L ...

  9. jmeter--响应断言

    背景 在测试过程中,我们需要对某个请求的结果进行判定. 比如我们搜索“你好”,请求发送成功,返回响应码也是200,但是并不能说明返回的响应就是对的,我们可能还需要判定响应结果包含“你好”.这个时候,我 ...

  10. 【Codeforces Round #299 (Div. 2) C】 Tavas and Karafs

    [链接] 我是链接,点我呀:) [题意] 给你一个规则,让你知道第i根萝卜的高度为si = A+(i-1)*B 现在给你n个询问; 每次询问给你一个固定的起点l; 让你找一个最大的右端点r; 使得l. ...