Linux(编程-进程基础):08:---进程创建(fork、vfork)

一、fork()

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

//返回:子进程返回0,父进程返回子进程ID,出错为-1

1.功能

  • fork可用于创建一个子进程

2.fork的返回值

fork函数被调用一次,但返回两次,成功时:

  • 子进程的PID将在父进程中返回
  • 而0将在子进程中返回

失败时:

  • -1:是在父进程中返回,不创建子进程,并且正确设置errno

3.创建的子进程与父进程之间的关系

  • ①子进程是父进程的副本。例如,子进程获得父进程数据空间、堆和栈的副本
  • ②因为子进程是副本。所以子进程只是拷贝父进程的内容,但是父、子进程并不共享这些存储空间部分
  • ③父、子进程共享正文段

子进程继承父进程的其他性质

  • 实际用户ID、实际组ID、有效用户ID、有效组ID、添加组ID、进程组ID、对话期I D
  • 控制终端、设置-用户-ID标志和设置-组-ID标志、当前工作目录、根目录、文件方式创建屏蔽字、信号屏蔽和排列、对任一打开文件描述符的在执行时关闭标志、环境、连接的共享存储段、资源限制

父、子进程之间的区别

  • fork的返回值、进程ID、不同的父进程ID
  • 子进程的s tms_utime ,tms_stime , tms_cutime, 以及tms_cstime设置为0
  • 父进程设置的锁子进程不继承
  • 子进程的未处理闹钟被清除
  • 子进程的未处理信号集设置为空集

4.fork的两种用法

  • 一个父进程希望复制自己,使父、子进程同时执行不同的代码段。这在网络服务进程 中是常见的——父进程等待委托者的服务请求。当这种请求到达时,父进程调用fork,使子进程处理此请求。父进程则继续等待下一个服务请求
  • 一个进程要执行一个不同的程序。这对shell是常见的情况。在这种情况下,子进程在从fork返回后立即调用e x e

5.写时复制

  • 由于在fork之后进程跟随着exec,所以现在的很多实现并不执行一个父进程数据段、栈和堆的完全副本。作为替代,使用了写时复制(copy-on-write ,COW)技术
  • 这些区域由父、子进程共享,而且内核将它们的访问权限改变为只读
  • 如果父、子进程中的任一试图修改这些区域,则内核只为修改区域的那块内存制作一个副本,通常是虚拟存储系统的一“页”

6.案例

int globvar = 6;
char buf[] = "a write to stdout\n";
int main(void)
{
    int var; 
    pid_t pid;
    var = 88;
    if (write(STDOUT_FILENO, buf, sizeof(buf)-1) != sizeof(buf)-1)
        perror("write error");
    printf("before fork\n"); 
    if ((pid = fork()) < 0) {
        perror("fork error");
    } 
    else if (pid == 0) { 
        globvar++;
        var++;
    } 
    else {
        sleep(2); 
    }
    printf("pid = %ld, glob = %d, var = %d\n", (long)getpid(), globvar,var);
    exit(0);
}

实验一:

  • 先打印信息。
  • 然后打印子进程的pid,再打印父进程的pid(因此父、子进程的执行时随机的,先后不确定。所以程序让父进程sleep3秒,让子进程先执行)

实验二:

  • 我们让输出信息重定向到一个文件中

"before fork"打印两次的原因:

  • 标准输出如果输出到终端设备,则是行缓冲,所以实验一只打印一次
  • 而实验二不是输出到终端设置,则是全缓冲。因为printf还没有打印,子进程创建了,此时缓冲区的数据就会复制一份到子进程中。因此当每个进程终止时,其缓冲区中的内容都被写到文件中

7.父、子进程文件共享

fork的一个特性是所有由父进程打开的描述符都被复制到子进程中

  • 这种共享文件的方式使父、子进程对同一文件使用了一个文件位移量。因此父、子进程对同一文件描述符操作时会出现混乱

在fork之后后处理文件描述符有两种常见的情况:

  • 父进程等待子进程完成。在这种情况下,父进程无需对其描述符做任何处理。当子进程终止后,它曾进行过读、写操作的任一共享描述符的文件位移量已做了相应更新
  • 父、子进程各自执行不同的程序段。在这种情况下,在fork之后,父、子进程各自关闭它们不需使用的文件描述符,并且不干扰对方使用的文件描述符。这种方法是网络服务进程中 经常使用的

二、vfork()

#include <sys/types.h>
#include <unistd.h>
pid_t vfork(void);

//返回值:与fork相同

备注:

1.功能

  • vfork用于创建一个新进程,而该新进程的目的是exec一个新程序

2.vfork()的特点

  • 不过在子进程调用exec或exit之前,它在父进程的空间中运行。 这种工作方式在某些UNIX的实现中提高了效率
  • 但是如果子进程修改数据(除了用于存放vfork返回值的变量)、进行函数调用、或者没有调用exec或exit就返回都可能会带来未知的后果

3.与fork()的不同

  • vfork与fork一样都创建一个子进程, 但是它并不将父进程的地址空间完全复制到子进程中,因为子进程会立即调用exec(或exit),于 是也就不会存访该地址空间
  • 重点:vfork保证子进程先运行,在它调用exec或exit之后父进 程才可能被调度运行。(如果在调用这两个函数之前子进程依赖于父进程的进一步动作,则会导致死锁)

4.演示案例

int globvar = 6; 
int main(void)
{
    int var; 
    pid_t pid;
    var = 88;
    printf("before vfork\n"); 
    if ((pid = vfork()) < 0) {
        perror("vfork error");
    } 
    else if (pid == 0) { 
        globvar++;
        var++;
        _exit(0); 
    }

    printf("pid = %ld, glob = %d, var = %d\n", (long)getpid(), globvar,var);
    exit(0);
}

运行结果:

  • 因为vfork创建的子进程在父进程的地址空间中运行,所以vfrok()改变了值,父进程的也变化了

猜你喜欢

转载自blog.csdn.net/qq_41453285/article/details/88976501