进程控制之进程标识、函数fork、vfork以及exit

进程标识

每个进程都有一个非负整型表示的唯一进程ID。ID为0的进程通常是调度进程,常常被称之为交换进程,该进程是内核的一部分,它并不执行任何磁盘上的程序,因此也被称之为系统进程。进程ID为1的通常是init进程,在自举过程结束时由内核调用。该进程的程序文件在UNIX的早期版本中是/etc/init,在较新版本中是/sbin/init。此进程负责在自举内核后启动一个UNIX系统。

除了进程ID之外,每个进程还有一些其他的标识符。

#include <unistd.h>
pid_t getpid(void);
返回值:调用进程的进程ID。
pid_t getppid(void);
返回值:调用进程的父进程ID。
uid_t getuid(void);
返回值:调用进程的实际用户ID。
uid_t geteuid(void);
返回值:调用进程的有效用户ID。
gid_t getgid(void);
返回值:调用进程的实际组ID。
gid_t getegid(void);
返回值:调用进程的有效组ID。

测试示例:

#include "../../include/apue.h"

int main()
{
    pid_t pid, ppid;
    uid_t uid, euid;
    gid_t gid, egid;
    
    printf("pid = %d, ppid = %d, uid = %d, euid = %d, gid = %d, egid = %d\n", 
           pid = getpid(), ppid = getppid(), uid = getuid(), euid = geteuid(), 
           gid = getgid(), egid = getegid());

    return 0;
}

结果如下:

函数fork

现有进程可以通过调用函数fork来创建一个新进程(即子进程)。

#include <unistd.h>
pid_t fork(void);
返回值: 子进程返回 0, 父进程返回子进程ID;若出错,返回 -1

一次fork函数的调用,返回两次。两次返回的区别是子进程的返回值是0,而父进程的返回值则是新建子进程的进程ID。子进程是父进程的副本。

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

测试示例:

#include "../../include/apue.h"
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)
        err_sys("write error");
    printf("before fork\n");

    if((pid = fork()) < 0){
        err_sys("fork error");
    }else if(pid == 0){
        globvar++;
        var++;
        printf("child pid = %ld, glob = %d, var = %d\n", (long)getpid(), globvar, var);
    }else{
        sleep(2);
        printf("parent pid = %ld, glob = %d, var = %d\n", (long)getpid(), globvar, var);
    }

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

结果如下:

除了打开文件的共享之外,父进程的很多其他属性也由子进程继承,这些属性如下:

  • 实际用户ID、实际组ID、有效用户ID、有效组ID。
  • 附属组ID。
  • 进程组ID。
  • 会话ID。
  • 控制终端。
  • 设置用户ID标志和设置组ID标志。
  • 当前工作目录。
  • 根目录。
  • 文件模式创建屏蔽字。
  • 信号屏蔽和安排。
  • 对任一打开文件描述符的执行时关闭标志。
  • 环境。
  • 连接的共享存储段。
  • 存储映像。
  • 资源限制。

父进程和子进程之间的区别。

  • fork的返回值不同。
  • 进程ID不同。
  • 这两个进程的父进程ID不同:子进程的父进程ID是创建它的进程的ID,而父进程的父进程ID则不变。
  • 子进程的tms_utimetms_stimetms_cutimetms_ustime的值设置为0。
  • 子进程不继承父进程设置的文件锁。
  • 子进程的未处理闹钟被清除。
  • 子进程的未处理信号集设置为空集。

fork的两种用法:

  1. 一个父进程希望复制自己,是父进程和子进程同时执行不同的代码段。
  2. 一个进程要执行一个不同的程序。

函数vfork

vfork函数用于创建一个新进程,而该新进程的目的是执行一个新程序。vforkfork函数一样都是创建一个子进程,但是它并不将父进程的地址空间完全复制到子进程中,因为子进程会立即调用exec(或exit),于是就不会引用该地址空间。

vforkfork之间的区别是:vfork保证子进程先运行,在它调用execexit之后父进程才可能被调度运行,当子进程调用这两个函数中的任意一个时,父进程会恢复运行。(如果在调用这两个函数之前子进程依赖于父进程的进一步动作,则会导致死锁。)

测试示例:

#include "../../include/apue.h"
int globvar = 6;

int main(void)
{
    int var;
    pid_t pid;

    var = 88;
    printf("before vfork\n");

    if((pid = vfork()) < 0){
        err_sys("fork error");
    }else if(pid == 0){
        globvar++;
        var++;
        exit(0);
        //_exit(0);
    }

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

结果如下:

函数exit

进程有5种正常终止及3种异常终止的方式。

5种正常终止的方式如下:

  1. main函数内执行return语句。
  2. 调用exit函数。
  3. 调用_exit_Exit函数。大多数UNIX系统实现中,exit是标准C库中的一个函数,而_exit则是一个系统调用。
  4. 进程的最后一个线程在其启动例程中执行return语句。
  5. 进程的最后一个线程调用pthread_exit函数。

3种异常终止如下:

  1. 调用abort函数。它产生SIGABRT信号。
  2. 当进程接收到某些信号时。信号可由进程本身、其他进程或内核产生。
  3. 最后一个线程对“取消”请求做出响应。

如果父进程在子进程之前终止,对于父进程已经终止的所有进程而言,它们的父进程都变为init进程。具体过程是在一个进程终止时,内核会检查所以活动的进程,以判断它是否是正要终止进程的子进程,如果是,则该进程的父进程ID就更改为1。这种处理方法保证了每个进程有一个父进程。在UNIX术语中,一个已经终止、但是其父进程尚未对其进行善后处理的进程被称为僵尸进程(zombie)。ps命令查看下将其表示为Z状态。

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值