一般我们启动一个进程都是通过 路径+程序名 启动,这是由用户决定,速度比较慢;为了提高速度,我们可以通过一个进程创建另一个进程,这是由程序决定,所以在这里我们可以使用fork()方法创建进程。
fork()方法: 返回值pid_t类型(实质是int类型)
pid_t fork(void); //创建进程
fork方法会新生成一个进程,调用fork方法的进程为父进程,新生成的进程是子进程。
在父进程中返回子进程的pid,在子进程中返回0,失败返回-1。也就是说fork()方法是复制进程,但不复制pid,如果pid=0,处于子进程;如果pid != 0,处于父进程。所以在执行上述代码时,父进程执行的是else语句,输出world;子进程执行if语句,输出hello。
fork()方法调用后,父、子进程都从fork调用之后的指令开始执行。
pid_t getpid(); //获取当前进程的pid
pid_t getppid(); //返回当前进程父进程的pid
fork()方法调用后,父子进程的关系: 并发执行
示例:
示例1:
for(i=0;i<2;++i)
{
pid_t pid=fork();
assert(pid!=-1);
if(pid==0)
{
printf("A\n");
}
else
{
printf("B\n");
}
}
输出3个B,3个A
示例2:去掉AB后面的\n,数据进入缓冲区
for(i=0;i<2;++i)
{
pid_t pid=fork();
assert(pid!=-1);
if(pid==0)
{
printf("A");
}
else
{
printf("B");
}
}
输出4个A,4个B,因为缓冲区的数据没有打印,一起被复制了
- malloc申请空间并不是malloc成功后就直接将物理内存空间分配给用户,而是在用户使用的时候才会给用户分配物理内存空间。malloc调用成功只是将虚拟地址空间上的堆区空间分配给用户。
- fork方法并不会直接将父进程的数据空间复制给子进程,而是子进程在修改数据空间上的数据时,才会给子进程分配空间。(写时拷贝技术)
- 释放空间时,会直接将物理内存空间释放。
- fork之后,刚开始,父子进程共享父进程的数据段、栈和堆。
fork之后父子进程对于全局数据,局部数据和堆区空间用的不是同一块空间,子进程修改值不会影响到父进程