欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页

Unix环境高级编程笔记:8、进程控制

程序员文章站 2022-03-02 15:49:19
...
1、进程标识符
    每个进程都有一个非负整型表示的唯一进程ID。
    ID为0的进程通常是调度进程,常常被称为交换进程(swapper)。
    ID为1通常是init进程,在自举过程结束时由内核调用。该进程文件在/etc/init
 
    #include <unistd.h>
    gid_t getegid(void); 获取有效用户ID
    uid_t geteuid(void);
    git_t getgid(void);
    uid_t getuid(void);  获取真实用户ID
 
2、fork
    #include <unistd.h>
    pid_t fork(void);        返回值:子进程中返回0,父进程中返回子进程ID,出错返回-1
    fork函数被调用一次,但返回2次,二次返回的唯一区别是子进程返回值是0,而父进程的返回值是新子进程的进程ID。 
 
    子进程和父进程继续执行fork调用之后的指令。子进程是父进程的副本。例如,子进程获得父进程数据空间、堆和栈的副本。
    父、子进程并不共享这些存储空间部分,父、子进程共享正文段。
 
#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
 
int glob = 6;
char buf[] = "a write to stdout\n";
 
int main(int argc, char **argv) {
int var;
pid_t pid;
 
var = 88;
 
if(write(STDOUT_FILENO,buf,sizeof(buf)-1) != sizeof(buf)-1) {
fprintf(stdout,buf);
exit(1);
}
printf("before fork");
 
if((pid = fork()) <0) {
printf("fork error");
} if(pid == 0) {
glob++;
var ++;
} else {
sleep(2);
}
printf("pid = %d,glob = %d,var = %d\n",getpid(),glob,var);
exit(0);
}
 
 
3、wait和waitpid函数
    当一个进程正常或异常终止时,内核就向其父进程发送SIGCHLD信号。因为子进程终止是个异步事件。
 
    调用wait或waitpid进程可能发生什么情况:
    a)如果其所有子进程都还在运行,则阻塞。
    b)如果一个子进程已经终止,正等待父进程获取其终止状态,则取得该子进程的终止状态立即返回。
    c)如果它没有任何子进程,则立即返回
 
    #include<sys/wait.h>
    pid_t wait(int *statloc);
    pid_t waitpid(pid_t pid,int *statloc,int options);
    返回值:成功返回进程ID,或出错则返回-1
    两个函数的区别:
    a)在一个子进程终止前,wait使其调用者阻塞,而waitpid有一个选项,可使调用者不阻塞。
    b)waitpid并不等待在其调用之后的第一个终止子进程,它有若干个选项,可以控制它所等待的进程。
 
    这二个函数的参数statloc是一个整形指针,如果statloc不是一个空指针,则终止进程的终止状态就存放在它所指向的单元内。如果不关心终止状态,则可以
    将该参数指定为空指针。
 
    检查wait和waitpid所返回的终止状态的宏
    WIFEXITED(status)
    WIFSIGNALED(status)
    WIFSTOPPED(status)
    WIFCONTINUED(status)
 
    打印进程终止状态
    
#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
#include <sys/wait.h>
 
void pr_exit(int status) {
if (WIFEXITED(status)) {
printf("normal termination,exit status   %d\n", WEXITSTATUS(status));
} else if (WIFSIGNALED(status)) {
printf("abnormal termination,signal number = %d%s\n", WTERMSIG(status),
#ifdef WCOREDUMP
WCOREDUMP(status) ? "(core file generated)" : "");
#else
"");
#endif
} else if(WIFSTOPPED(status)) {
printf("child stopped,signal nuber=%d\n",WSTOPSIG(status));
}
}
 
int main(int argc, char **argv) {
pid_t pid;
int status;
if((pid = fork()) <0) {
printf("fork error");
} else if(pid == 0) {
exit(7);
}
 
if(wait(&status) != pid) {
printf("wait");
}
pr_exit(status);
 
exit(0);
}
 
    
如果一个进程有几个子进程,那么只要一个子进程终止,wait就返回。
 
waitpid函数返回终止子进程的进程ID,并将该子进程的终止状态存放在由statloc指向的存储单元中。
pid==-1        等待任一子进程,就这一方面而言,waitpid与wait等效。
pid > 0          等待其进程ID与pid相等的子进程。
pid == 0        等待其组ID等于调用进程组ID的任一子进程。
pid < -1        等待其组ID等于pid绝对值的任一子进程。
 
常量 说明
WCONTINUED
 
WHOHANG
 
WUNTTRACED
若实现支持作业控制,那么由pid指定的任一子进程在暂停后已经继续,但其状态尚未报告,则返回其状态
 
若由pid指定的子进程并不是立即可用的,则waitpid不阻塞,此时其返回0
 
若某实现支持作业控制,而由pid指定的任一子进程处理于暂停状态,并且其状态自暂停以来还未报告过,则返回
其状态
 
 
    waitpid函数区别于wait
    a)waitpid可等待一个特定的进程,而wait则返回任一终止子进程的状态
    b)waitpid提供了一个wait的非阻塞版本
    c)waitpid支持作业控制
 
 
4、waitid
5、wait3 wait4
 
6、竞争条件
    当多个进程都企图对共享数据进行某种处理,而最后的结果又取决于进程运行的顺序时,则我们认为这发生了竟争条件(race condition)
 
7、exec函数
    当进程调用exec函数时,该 进程执行的程序完全替换为新程序,而新程序则从其main函数开始执行。
    因为调用exec并不创建新进程,所以前后进程的进程ID并未改变。
    exec只是用一个全新的程序替换了当前进程的正文、数据、堆和栈段。
 
    用fork可以创建新进程,用exec可以执行新程序。exit函数和二个wait函数处理终止和等待终止。这些是我们需要的基本的进程控制原语
 
    
#include <unistd.h>
execl
execv
execle
execve
execlp
execvp
返回值 :若出错则返回-1,若成功则不返回值
 
8、更改用户ID和组ID
    在UNIX系统中,特权是基于用户和组ID的。
    当程序需要增加特权,或需要访问当前并不允许访问的资源时,我们需要更换自己的用户ID或组ID,使得新ID具有合适的特权或访问权限。
 
    在设计应用程序时,我们总是试图使用最小特权模型。   
#include <unistd.h>
setuid
setgid
 
9、system   
#include <stdlib.h>
int system(const char *cmdstring);
 
 
10、用户标识
#include <unistd.h>
char *getlogin(void);
 
找到运行该程序的用户登录名:可以调用 getpwuid(getuid())
 
getpwnam 在口令文件中查找用户的相应记录,确定其登录shell