1. 简介
本章介绍UINIX系统的进程控制,包括创建新进程,执行程序和进程终止。还将说明进程属性的各种ID-----实际、有效和保存的用户ID和组ID。
2.函数原型
pid_t getpid(void); pid_t getppid(void); | 返回调用进程和调用进程父进程的进程ID |
uid_t getuid(void); uid_t geteuid(void); | 返回调用进程的实际用户ID和有效用户ID |
gid_t getgid(void); gid_t getegid(void); | 返回调用进程的实际组ID和有效组ID |
pid_t fork(void); | 创建一个新进程 返回值:子进程返回0,父进程返回子进程ID;若出错,返回-1 |
pid_t vfork(void); | 创建一个新进程,保证子进程先运行,返回值与fork相同 fork和vfork最大区别是vfork在调用exec(或exit)之前共享父进程的数据、堆和栈空间,因此如果子进程运行不立即调用exec(或exit),可能会产生未知的结果 |
1. pid_t wait(int *wstatus); 2. pid_t waitpid(pid_t pid, int *wstatus, int options); 3. int waitid(idtype_t idtype, id_t id, siginfo_t *infop, int options); 4. pid_t wait3(int *wstatus, int options,struct rusage *rusage); 5. pid_t wait4(pid_t pid, int *wstatus, int options, struct rusage *rusage); | 等待进程改变状态 返回值:若成功,返回进程ID;若出错,返回0或-1 |
1. int execl(const char *pathname, const char *arg, ...); 2. int execlp(const char *file, const char *arg, ...); 7. int fexecve(int fd, char *const argv[], char *const envp[]); | 执行一个新程序,替换当前进程的正文段、数据段、堆段和栈段 返回值:若成功,不返回;若出错,返回-1 |
int setuid(uid_t uid); int setgid(gid_t gid); | 设置用户ID和组ID 返回值:若成功,返回0;若出错,返回-1 |
int setreuid(uid_t ruid, uid_t euid); int setregid(gid_t rgid, gid_t egid); | 设置实际和(或)有效用户ID和组ID 返回值:若成功,返回0;若出错,返回-1 |
int seteuid(uid_t euid); int setegid(gid_t egid); | 设置有效用户和组ID 返回值:若成功,返回0;若出错,返回-1 |
int system(const char *command); | 执行一个shell命令 返回值:若成功,返回shell终止状态;若失败返回-1或127 |
char *getlogin(void); | 返回值:若成功,返回登录名;若失败,返回NULL |
int nice(int inc); | 改变进程调度优先级 返回值:若成功,返回新的nice值NZERO;若出错,返回-1 |
int getpriority(int which, id_t who); int setpriority(int which, id_t who, int prio); | 获取/设置进程调度优先级 |
clock_t times(struct tms *buf); | 获取进程时间 返回值:若成功,返回流逝的墙上时钟时间;若出错,返回-1 |
3. 实例
#include <stdio.h>
#include <unistd.h>
#include <errno.h>
#include <string.h>
#include <stdlib.h>
#include <sys/wait.h>
char *argv[] = {"ls", "/usr/", "/proc", NULL};
int main()
{
printf(" pid: %d\n", getpid());
printf("ppid: %d\n", getppid());
printf(" uid: %d\n", getuid());
printf("euid: %d\n", geteuid());
printf(" gid: %d\n", getgid());
printf("egid: %d\n", getegid());
int n = 10;
#if 0
pid_t pid = fork();
if (pid < 0) {
printf("fork error: %s\n", strerror(errno));
} else if (pid == 0) {
printf("I am the child process\n");
n++;
} else {
printf("I am the parent process\n");
}
#else
#if 1
pid_t pid = fork();
#else
pid_t pid = vfork();
#endif
if (pid < 0) {
printf("fork error: %s\n", strerror(errno));
} else if (pid == 0) {
printf("I am the child process\n");
n++;
int *sn = 100; // SIGSEGV==11
printf("sn: %d\n", *sn);
//_exit(128);
} else {
printf("I am the parent process\n");
int status;
waitpid(pid, &status, 0);
if (WIFEXITED(status)) {
printf("Normal termination: %d\n", WEXITSTATUS(status));
} else if (WIFSIGNALED(status)) {
printf("Abnormal termination: %d\n", WTERMSIG(status));
} else {
printf("Job control\n");
}
}
#endif
printf("n = %d\n", n);
pid_t pid2 = fork();
if (pid2 == 0) {
#if 0
if(execl("/usr/bin/ls", "ls", "/proc", "/mnt", NULL) < 0) {
printf("execl error\n");
}
#else
if(execvp("ls", argv) < 0) {
printf("execl error\n");
}
#endif
}
printf("pid2 = %d\n", pid2);
waitpid(pid2, NULL, 0);
return 0;
}
4. 小结
1. 熟练掌握fork、exec函数族、_exit、wait和waitpid这些函数
2. 理解各种不同的用户ID(实际、有效和保存)