- 创建子进程:fork函数
1. 函数原型:
int fork();
2. 函数作用:
一个进程的执行时,系统将分配地址空间给进程。
在进程执行过程中,如果调用fork函数创建一个子进程,子进程为父进程的拷贝。
二者用户区数据是一样的。
内核区中进程ID不一样。
3. 返回值
调用fork函数后,子进程拷贝父进程而创建,因此用户区代码是一样的。
在父进程中的fork返回值为子进程的ID;
子进程中的fork返回值为0.
4. 子进程创建成功后,代码执行位置
子进程的代码是父进程代码的拷贝,但是执行的位置是从父进程的fork函数调用之后的代码开始执行
5. 子进程创建成功后,父子进程执行的顺序
进程的执行顺序是由CPU决定的,因此不能保证执行顺序。
由于CPU中是轮循执行程序的,因此不能由屏幕的输出顺序来决定执行顺序。
6. 终端显示错误
程序执行时在前台,shell进程进入后台。
由于shell不知道程序中是否创建了子进程,若父子进程中父进程先执行完,shell就会切换回前台显示,而子进程仍在执行。
如果子进程执行完毕后,父进程再执行完毕,则不会出现终端显示错误。
- 获取进程PID:getpid和getppid
1. 函数原型:
pid_t getpid();
pid_t getppid();
2. 返回值:
getpid:返回当前进程的PID
getppid:返回当前进程的父进程PID
- 循环创建子进程
1. 由于子进程拷贝父进程内容,在循环创建子进程时,需要判定子进程pid进行break操作
否则,子进程也会因为循环而在其自身上创建子进程
for (int i = 0; i < num; i++)
{
pid = fork();
if (pid == 0)
break;
}
- 进程间的数据共享
1. 刚开始fork时,两个地址空间用户区数据完全相同
2. 但是,两个进程开辟的空间位置是不同的,因此,父进程和子进程的数据是完全独立的。
3. 当数据只有读操作时,父子进程的同名变量在物理内存中指向同一内存(为了节约内存)
4. 当数据在父子进程中发生写操作时,在内存中会复制一份原变量值,然后修改拷贝后的变量,并不会直接修改原地址的值。
5. 即:读时共享,写时复制
##Tip##
因此,父子进程之间不能通过全局变量来通信。