系统级程序设计 第六章
进程管理
课前复习
上节课老师为我们介绍了基本的文件操作内容,由于时间原因还有一小部分内容没有完成,于是乎,在开启新的章节之前,让我们先将文件操作的内容加以完善!
1.1文件操作
1.1.1stat()函数
#include <sys/stat.h>
int stat(const char *path, struct stat *buf);
功能:用于获取文件属性;
参数说明:第一个参数path为文件路径
第二个参数buf用于接收获取到的文件属性
注意:文件的属性存储于inode中,因此stat()函数实际上是从inode结构体中获取文件信息
返回值说明:
成功:0
不成功:-1并设置errno
测试样例:
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <sys/stat.h>
//使用stat()函数获取“a.out"文件属性,并输出文件大小
int main(){
struct stat sbuf;
int tempRet = 2;
tempRet = stat("a.out",&sbuf);
if(tempRet==-1){
perror("stat error:");
exit(1);
}//of if
printf("len = %ld\n",sbuf.st_size);
return 0;
}//of main
运行结果:
1.1.2access()函数
#include <unistd.h>
int access(const char *pathname, int mode);
功能:用于测试文件是否拥有某种权限
参数说明:
pathname:文件名;
mode:取值有4个:R_OK, W_OK, X_OK, F_OK;前面3个是测试文件是否有读、写、执行权限,最后一个测试文件是否存在。
返回值说明:
成功:0,表示文件存在或具有某种权限。
不成功:-1并设置errno
1.1.3chmod()函数
#include <sys/stat.h>
int chmod(const char *path, mode_t mode);
功能:用于修改文件的访问权限
参数说明:
path:路径名
mode:用于传递修改后的权限
返回值说明:
成功:返回0
不成功:返回-1,并设置errno值
1.1.4truncate()函数
#include <sys/stat.h>
int truncate(const char *path, off_t length);
功能:用于修改文件大小,常用于扩展文件,其功能与lseek函数类似
参数说明:
path:路径名
length:设置文件大小
返回值说明:
成功:0;
不成功:-1并设置errno。
至此,上一章内容已经全部完结,接下来就是本文正题:进程管理
2.1进程管理
2.1.1 进程概述
进程是一个二进制程序的执行过程,在linux操作系统中,向命令行输入一条命令,按下回车,便会有一个进程启动。
2.2.2进程属性
1.标识符
1)进程标识符:即进程ID,简称pid,他是进程的唯一标识;内核通过这个标识来识别不同的进程;用户也可以根据内核提供的pid,通过系统调用去操作用户进程。
2)父进程标识符:简称ppid,是进程的父进程,即创建该进程的进程所对应的pid。
2.2.3创建进程
1.fork()函数
#include <unistd.h>
pid_t fork(void);
//调用fork()函数创建的进程称为子进程,调用fork()函数的进程为父进程
功能:创建进程;函数执行后,系统会创建一个与原进程几乎相同的进程,之后父子进程都继续执行。
参数说明:无
返回值说明:
成功:返回两个值,子进程创建成功后,原程序会被复制,就有了两个fork函数。父进程的fork函数会返回子进程的pid,子进程的fork函数会返回0.
不成功:若子进程创建失败,原程序不会复制,父进程的fork函数返回-1。
测试样例:
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
int main(){
pid_t tempPid;
tempPid = fork();//调用fork()函数创建子进程
if(tempPid == -1){//创建失败
perror("fork error");
}else if(tempPid > 0){//父进程
printf("parent process, pid = %d, ppid = %d\n", getpid(), getppid());
}else if(tempPid==0){//子进程
printf("child process, pid = %d, ppid = %d\n", getpid(), getppid());
}//of if
printf("......finish......\n");
return 0;
}//of main
运行截图:
这里发现一个问题,子进程的父进程id与父进程id不同,而且也不为1,这个问题在切换到字符界面运行就会发现子进程的ppid为1
思考:多次执行test_fork会发现,child process后输出的ppid不等于parent process的pid,而等于1。请说明原因。
解释:
出现这种情况,是因为父进程先于子进程结束,此时,子进程并未终止,父进程结束,子进程就变成了孤儿进程。
孤儿进程是无法关闭的,因为其没有父进程控制,
但linux将孤儿进程分配给init进程,于是会出现ppid=1的情况
2.2.4创建多个进程
测试样例:
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
int main(){
pid_t tempPid;
int i;
for(i = 0; i < 5; i ++){
if((tempPid = fork()) == 0){
break;
}//of if
}//of for i
if(tempPid == -1){
perror("fork error");
}else if(tempPid > 0){//parent
printf("parent process, pid = %d, ppid = %d\n", getpid(), getppid());
}else{//child
printf("I am child process = %d, pid = %d, ppid = %d\n", i + 1, getpid(), getppid());
}//of if
printf("......finish......\n");
return 0;
}//of main
运行截图:
1.这里ppid的问题个人猜测原因和测试1一样
2.为什么pid排列没有顺序?
答:因为这些子进程属于同一优先级,获得的系统资源也一样,先获得的就先输出。
3.终端提示符后面仍然有子进程信息打印,而命令提示符在最后一行的开头闪烁。这是为什么?
答:同样从优先级考虑,因为父进程、子进程还有bash命令优先级相同,共同竞争系统资源,CPU随机调用,谁优先获取谁就先执行。
2.2.5进程的执行顺序
利用sleep()函数暂缓进程执行
测试:
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
int main(){
pid_t tempPid;
int i;
for(i = 0; i < 5; i ++){
if((tempPid = fork()) == 0){
break;
}//of if
}//of for i
if(tempPid == -1){
perror("fork error");
}else if(tempPid > 0){//parent
sleep(2);
printf("parent process, pid = %d, ppid = %d\n", getpid(), getppid());
}else{//child
sleep(i);
printf("I am child process = %d, pid = %d, ppid = %d\n", i + 1, getpid(), getppid());
}//of if
printf("......finish......\n");
return 0;
}//of main
运行截图:
这里需要注意一个问题,要注意父进程的阻塞时间设置要长于子进程,否则后面的子进程就会变成孤儿进程。
总结:
有些时候bash命令会抢先于子进程出现,并且后续才会继续创建子进程;原因是父进程、子进程、bash命令进程优先级相同,当bash命令优先抢占成功便会出现以上情况。