(1)fork()的定义
fork()函数是Unix中派生新进程的唯一方法,声明如下:
- #include <unistd.h>
- pid_t fork(void);
我们需要理解的是,调用一次fork()方法,该方法会返回两次。一次是在调用进程(也就是派生出的子进程的父进程)中返回一次,返回值是新派生的进程的进程ID。一次是在子进程中返回,返回值是0,代表当前进程为子进程。如果返回值为-1的话,则代表在派生新进程的过程中出错。
那么在程序中,我们就可以根据此返回值来判断当前进程是父进程还是子进程,来实现一些具体的操作。例如:
- int main()
- {
- pid_t pid;
- if((pid = fork()) = 0)
- {
- // TODO: 在子进程中实现具体操作
- // ...
- exit(0); // 结束子进程
- }
- // TODO: 在调用进程(父进程)实现具体操作
- }
(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;
}
}
}