天天看点

《APUE》读书笔记—第八章进程控制

本章介绍unix的进程控制,包括进程创建,执行程序和进程终止,进程的属性,exec函数系列,system函数,进程会计机制。

1、进程标识符

  每一个进程都有一个非负整数标识的唯一进程id。id为0表示调度进程,即交换进程,是内核的一部分,也称为系统进程,不执行任何磁盘操作。id为1的进程为init进程,init进程不会终止,他是一个普通的用户进程,需要超级用户特权运行。获取标识符函数如下:

#include <sys/types.h>

#include <unistd.h> 

pid_t getpid(void);    //调用进程的进程id

pid_t getppid(void);     //调用进程的父进程id

gid_t getgid(void);       //调用进程的实际组id

gid_t getegid(void);    //调用进程的有效组id

uid_t getuid(void);     //调用进程的实际用户id

uid_t geteuid(void);   //调用进程的有效用户id

2、fork函数

  一个现有的进程可以调用fork函数创建一个新进程。函数原型为:pid_t fork(void)。有fork创建的新进程称为子进程,fork函数调用一次返回两次。子进程的返回值为0,父进程的返回值为新子进程的id。子进程和父进程举行执行fork调用后的指令,子进程是父进程的副本。子进程获得父进程的数据空间、栈和队的副本,父子进程不共享这些存储空间部分,共享正文段。子进程相当于父进程克隆了一份自己。 创建新进程成功后,系统中出现两个基本完全相同的进程,这两个进程执行没有固定的先后顺序,哪个进程先执行要看系统的进程调度策略。

  fork出错可能有两种原因:

    1)当前的进程数已经达到了系统规定的上限,这时errno的值被设置为eagain。

    2)系统内存不足,这时errno的值被设置为enomem。

注意的是:父进程设置的文件锁不会被子进程继承。

写个程序,创建一个子进程,在子进程中改变变量的值,然后父子进程同时输出变量的值,看看有什么变化,同时输出子进程的标识符信息。程序如下:

《APUE》读书笔记—第八章进程控制
《APUE》读书笔记—第八章进程控制
《APUE》读书笔记—第八章进程控制

 程序执行结果如下:

《APUE》读书笔记—第八章进程控制

从结果可以看出子进程拥有自己的数据空间,不与父进程共享数据空间。

  vfork函数的调用序列和返回值与fork相同,但是vfork并不将父进程的地址空间完全复制到子进程中,在调用exec和exit之前在父进程的空间中运行。vfork保证子进程先运行,在它调用exec或exit之后父进程才可能被调度运行。

  进程终止最后都会执行内核中的同一段代码,为相应的进程关闭所有打开描述符,释放它所使用的存储器等。子进程可以通过exit函数通知父进程是如何终止的,父进程调用wait或waitpid函数可以获取终止状态。子进程是在父进程调用fork后产生的,如果父进程在子进程之前终止,则将子进程的父进程改变为init进程,保证每一个进程都有一个父进程。一个已经终止,但其父进程尚没有对其进行善后处理的进程称为僵死进程(zombie)。由init进程领养的子进程不会变成僵死进程,因为init进程在子进程终止的时候会调用一个wait函数取得子进程的终止状态。

  当一个进程正常或者异常终止的时,内核就像其父进程发送sigchld信号。调用wait和waitpid函数可能发生的情况:(1)如果所有子进程都还在运行,则阻塞;(2)如果一个子进程已经终止,正等待父进程获取进程终止状态,则取得孩子的终止状态立刻返回;(3)若果没有任何子进程,则立即出错返回。如果在任意时刻调用wait,则进程可能会阻塞。

pid_t wait(int *status);   //在一个子进程终止前,wait使调用者阻塞

pid_t waitpid(pid_t pid, int *status, int options); //可以使调用者不阻塞

pid_t wait3(int *status, int options,struct rusage *rusage);

pid_t wait4(pid_t pid, int *status, int options, struct rusage *rusage);

写个程序,子进程给出退出状态,父进程通过wait和waitpid获取退出状态。程序如下:

《APUE》读书笔记—第八章进程控制
《APUE》读书笔记—第八章进程控制
《APUE》读书笔记—第八章进程控制

程序执行结果如下:

《APUE》读书笔记—第八章进程控制

waitpid函数中的pid参数取值情况:

pid=-1   等待任一子进程,此时相当于wait

pid>0     等待期进程id与pid相等的子进程

pid==0    等待期组id等于调用进程组id的任一个子进程

pid<-1    等待其组id等于pid绝对值的任一子进程

另外提供了一种避免僵死进程的方法:调用fork两次。

  exec函数,用fork函数创建子进程后,子进程往往要调用一种exec函数以执行另外一个程序。当调用exec函数时,该进程执行的程序完全替换为新进程,exec函数不新建进程,只是用一个全新的程序替换了当前的正文、数据、堆和栈段。执行完之后,进程id不会改变。在进程间通信的时候,经常需要调用exec函数启动另外一个例程。函数原型如下:

extern char **environ; 

int execl(const char *path, const char *arg, ...);

int execlp(const char *file, const char *arg, ...);

int execle(const char *path, const char *arg,..., char * const envp[]);

int execv(const char *path, char *const argv[]);

int execvp(const char *file, char *const argv[]);

int execve(const char *file, char *const argv[],char *const envp[]);

其中l表示列表(list),v标识矢量(vector)。execl、execlp、execle中每个命令行参数都是一个单独参数,这种参数以空指针结尾。execv、execvp、execve命令行参数是一个指针数组。e标识环境变量,传递参数。写个程序进行测试,程序分为两部分,exec调用的程序,exec执行程序。程序如下:

exec调用程序如下,可执行文件名称为exectest,存放在/home/anker/programs目录下。

《APUE》读书笔记—第八章进程控制
《APUE》读书笔记—第八章进程控制

 exec执行程序如下:存放在/home/anker/programs目录下。

《APUE》读书笔记—第八章进程控制
《APUE》读书笔记—第八章进程控制

执行结果如下:

《APUE》读书笔记—第八章进程控制

system函数,在程序中执行一个命令字符串。例如system("date > file")。函数原型如下:

int system(const char *command);、

system函数实现中调用了fork、exec和waitpid函数,因此有三种返回值。system函数实现一下,没有信号处理。程序如下:

《APUE》读书笔记—第八章进程控制
《APUE》读书笔记—第八章进程控制
《APUE》读书笔记—第八章进程控制

system函数可以设置用户的id,这是一个安全漏洞。

进程会计:启用后,每当进程结束时候内核就写一个会计记录,包括命令名、所使用的cpu时间总量、用户id和组id、启动时间等。accton命令启动会计处理,会计记录写到指定的文件中,linux中位于/var/account/ pacct。会计记录结构定义在<sys/acct.h>头文件中。

#define acct_comm 16

typedef u_int16_t comp_t;

struct acct {

    char ac_flag;           /* accounting flags */

    u_int16_t ac_uid;       /* accounting user id */

    u_int16_t ac_gid;       /* accounting group id */

    u_int16_t ac_tty;       /* controlling terminal */

    u_int32_t ac_btime;     /* process creation time(seconds since the epoch) */

    comp_t    ac_utime;     /* user cpu time */

    comp_t    ac_stime;     /* system cpu time */

    comp_t    ac_etime;     /* elapsed time */

    comp_t    ac_mem;       /* average memory usage (kb) */

    comp_t    ac_io;        /* characters transferred (unused) */

    comp_t    ac_rw;        /* blocks read or written (unused) */

    comp_t    ac_minflt;    /* minor page faults */

    comp_t    ac_majflt;    /* major page faults */

    comp_t    ac_swaps;     /* number of swaps (unused) */

    char      ac_comm[acct_comm+1];/* command name (basename of lastexecuted command; null-terminated) */

    char      ac_pad[x];    /* padding bytes */

};

用户标识,用getlogin函数获取用户的登录名。函数原型如下:char *getlogin(void)。

进程时间:墙上时钟时间、用户cpu时间和系统cpu时间。任一个进程都可以调用times函数获取它自己及已终止子进程时间。进程时间操作函数及结构如下:

clock_t times(struct tms *buf);

struct tms {

    clock_t tms_utime;  /* user time */

    clock_t tms_stime;  /* system time */

    clock_t tms_cutime; /* user time of children */

    clock_t tms_cstime; /* system time of children */

总结:通过本章的学习。完全的了解unix的进程控制,掌握了fork、exec簇、wait和waitpid进程控制函数。另外学习了system函数和进程会计。了解了解释器文件及其工作方式,用户标识和进程时间。

继续阅读