系统级程序设计(学习笔记)

系统级程序设计 第六章

进程管理

课前复习

上节课老师为我们介绍了基本的文件操作内容,由于时间原因还有一小部分内容没有完成,于是乎,在开启新的章节之前,让我们先将文件操作的内容加以完善!

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命令优先抢占成功便会出现以上情况。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值