Linux中的fork用法

(1)fork()的定义

fork()函数是Unix中派生新进程的唯一方法,声明如下:

 

  1. #include <unistd.h>  
  2.   
  3. pid_t fork(void);  


我们需要理解的是,调用一次fork()方法,该方法会返回两次。一次是在调用进程(也就是派生出的子进程的父进程)中返回一次,返回值是新派生的进程的进程ID。一次是在子进程中返回,返回值是0,代表当前进程为子进程。如果返回值为-1的话,则代表在派生新进程的过程中出错。

 

那么在程序中,我们就可以根据此返回值来判断当前进程是父进程还是子进程,来实现一些具体的操作。例如:

 

  1. int main()  
  2. {  
  3.     pid_t pid;  
  4.   
  5.     if((pid = fork()) = 0)  
  6.     {  
  7.         // TODO: 在子进程中实现具体操作  
  8.         // ...  
  9.         exit(0); // 结束子进程  
  10.     }  
  11.   
  12.     // TODO: 在调用进程(父进程)实现具体操作  
  13. }  

(2)fork()的实质过程

 

父进程中在调用fork()派生新进程,实际上相当于创建了进程的一个拷贝;即在fork()之前的进程拥有的资源会被复制到新的进程中去。网络服务器在处理并发请求时,也可以采取这种派生新进程的方式: 父进程调用accept()后调用fork()来处理每一个连接。那么,所接受的已连接的套接口随后就在父子进程中共享。通常来说,子进程会在这连接套接口中读和写操作,父进程则关闭这个已连的套接口(可以参考:http://blog.csdn.net/moxiaomomo/article/details/6791763)

(3)fork()的用法

fork()有两个典型用法:(1)一个进程进行自身的复制,这样每个副本可以独立的完成具体的操作,在多核处理器中可以并行处理数据。这也是网络服务器的其中一个典型用途,多进程处理多连接请求。 (2)一个进程想执行另一个程序。比如一个软件包含了两个程序,主程序想调起另一个程序的话,它就可以先调用fork来创建一个自身的拷贝,然后通过exec函数来替换成将要运行的新程序。

fork调用的一个奇妙之处就是它仅仅被调用一次,却能够返回两次,它可能有三种不同的返回值:
    1)在父进程中,fork返回新创建子进程的进程ID;
    2)在子进程中,fork返回0;
    3)如果出现错误,fork返回一个负值;

创建新进程成功后,系统中出现两个基本完全相同的进程,这两个进程执行没有固定的先后顺序,哪个进程先执行要看系统的进程调度策略。此时,两个进程都从fork开始往下执行,只是pid不同。
https://blog.csdn.net/koches/article/details/7787468

 

 

 

分析下面一段代码:

#include <iostream>
#include <cstring>
#include <unistd.h>
using namespace std;

int main()
{
   int a = 1;
   int pid1 = fork();
   int pid2 = fork();
   if(pid1 == 0)
   {
      cout << "This is a child process" << endl;
      a = 2;
      cout << a << endl;
   }
   else
   {
      cout << "This is parent process" << endl;
      a = 3;
      cout << a << endl;
   }
   
}

 

输出如下:


This is a child process
2
This is a child process
2
This is parent process
3
This is parent process
3
[1] + Done                       "/usr/bin/gdb" --interpreter=mi --tty=${DbgTerm} 0<"/tmp/Microsoft-MIEngine-In-1row668y.g2k" 1>"/tmp/Microsoft-MIEngine-Out-b2nwu6sw.4rn"

Press any key to continue...

 

其实这里一共有4个进程,父进程A创建了子进程B和C,子进程B又创建了子进程D,其中父进程中pid1返回的是B进程的PID号,子进程B中返回0,由于fork是子进程对父进程资源的拷贝,所以在子进程C中,pid1是B进程的PID号,子进程D中的pid1是0,所以输出如上的结果。

总结:创建子进程-判断-使用-退出,防止出现僵尸进程,代码如下

#include <iostream>
#include <cstring>
#include <unistd.h>
#include <cstdlib>
using namespace std;

int main()
{
   int a = 1;
   int pid1 = fork();
   if(pid1 == 0)
   {
      cout << "This is a child process" << endl;
      a = 2;
      cout << a << endl;
      exit(0);
   }
   else
   {
      cout << "This is parent process" << endl;
      a = 3;
      cout << a << endl;
   }
   int pid2 = fork();
   if(pid2 == 0)
   {
      cout << "This is s child process" << endl;
      a = 4;
      cout << a << endl;
      exit(0);
   }
   
}

 

处理同一个任务的多个子进程

#include <iostream>
#include <cstring>
#include <unistd.h>
#include <cstdlib>
using namespace std;

int main()
{

   int a = 1;
   for(int i = 0; i < 2; ++i)
   {
      int pid = fork();
      if(pid == 0)
      {
         a = 2;
         cout << a << endl;
         exit(0);
      }
      else
      {
         a = 3;
         cout << a << endl;
      }
      
   }

}

 

  • 1
    点赞
  • 11
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值