广州大学学生实验报告
开课学院:计算机科学与网络工程学院
实验室:计算机软件实验室 2020 年5月20 日
学院 计算机科学与教育软件学院 年级/专业/班 计科183 姓名 张健介 学号 1806100162
实验课程名称 操作系统实验 成绩
实验项目名称 进程管理与进程通信 指导老师 张艳玲
实验一 进程管理与进程通信
一、实验目的
1、掌握进程的概念,明确进程的含义。
2、认识并了解进程并发执行的实质,进程的阻塞与唤醒,终止与退出的过程。
3、熟悉进程的睡眠、同步、撤消等进程控制方法。
4、分析进程竞争资源的现象,学习解决进程互斥的方法 。
5、了解什么是信号,利用信号量机制熟悉进程间软中断通信的基本原理,
6、熟悉消息传送的机理 ,共享存储机制 。
二、实验内容
1、编写一段程序,使用系统调用fork( )创建两个子进程。当此程序运行时,在系统中有一个父进程和两个子进程并发执行,观察实验结果并分析原因。
2、用fork( )创建一个进程,再调用exec( ),用新的程序替换该子进程的内容,利用wait( )来控制进程执行顺序,掌握进程的睡眠、同步、撤消等进程控制方法,并根据实验结果分析原因。
3、编写一段多进程并发运行的程序,用lockf( )来给每一个进程加锁,以实现进程之间的互斥,观察并分析出现的现象及原因。
4、编写程序:用fork( )创建两个子进程,再用系统调用signal( )让父进程捕捉键盘上来的中断信号(即按^c键);捕捉到中断信号后,父进程用系统调用kill( )向两个子进程发出信号,子进程捕捉到信号后分别输出下列信息后终止:
Child process1 is killed by parent!
Child process2 is killed by parent!
父进程等待两个子进程终止后,输出如下的信息后终止:
Parent process is killed!
分析利用信号量机制中的软中断通信实现进程同步的机理。
5、使用系统调用msgget( ),msgsnd( ),msgrev( ),及msgctl( )编制一长度为1k的消息发送和接收的程序,并分析消息的创建、发送和接收机制及控制原理。
6、编制一长度为1k的共享存储区发送和接收的程序,并设计对该共享存储区进行互斥访问及进程同步的措施,必须保证实现正确的通信。
三、实验原理
1、进程创建与进程并发执行
Linux中,进程既是一个独立拥有资源的基本单位,又是一个独立调度的基本单位。一个进程实体由若干个区(段)组成,包括程序区、数据区、栈区、共享存储区等。每个区又分为若干页,每个进程配置有唯一的进程控制块PCB,用于控制和管理进程。
系统为每个进程配置了一张进程区表。表中,每一项记录一个区的起始虚地址及指向系统区表中对应的区表项。核心通过查找进程区表和系统区表,便可将区的逻辑地址变换为物理地址。
进程是进程映像的执行过程,也就是正在执行的进程实体。它由三部分组成:
(1)用户级上、下文。主要成分是用户程序;
(2)寄存器上、下文。由CPU中的一些寄存器的内容组成,如PC,PSW,SP及通用寄存器等;
(3)系统级上、下文。包括OS为管理进程所用的信息,有静态和动态之分。
进程创建所涉及的系统调用:
fork( ) 创建一个新进程。
系统调用格式: pid=fork( )
参数定义:int fork( )
fork( )返回值意义如下:
0:在子进程中,pid变量保存的fork( )返回值为0,表示当前进程是子进程。
大于0:在父进程中,pid变量保存的fork( )返回值为子进程的id值(进程唯一标识符)。
-1:创建失败。
如果fork( )调用成功,它向父进程返回子进程的PID,并向子进程返回0,即fork( )被调用了一次,但返回了两次。此时OS在内存中建立一个新进程,所建的新进程是调用fork( )父进程(parent process)的副本,称为子进程(child process)。子进程继承了父进程的许多特性,并具有与父进程完全相同的用户级上下文。父进程与子进程并发执行。
核心为fork( )完成以下操作:
(1)为新进程分配一进程表项和进程标识符
进入fork( )后,核心检查系统是否有足够的资源来建立一个新进程。若资源不足,则fork( )系统调用失败;否则,核心为新进程分配一进程表项和唯一的进程标识符。
(2)检查同时运行的进程数目
超过预先规定的最大数目时,fork( )系统调用失败。
(3)拷贝进程表项中的数据
将父进程的当前目录和所有已打开的数据拷贝到子进程表项中,并置进程的状态为“创建”状态。
(4)子进程继承父进程的所有文件
对父进程当前目录和所有已打开的文件表项中的引用计数加1。
(5)为子进程创建进程上、下文
进程创建结束,设子进程状态为“内存中就绪”并返回子进程的标识符。
(6)子进程执行
虽然父进程与子进程程序完全相同,但每个进程都有自己的程序计数器PC(注意子进程的PC开始位置),然后根据pid变量保存的fork( )返回值的不同,执行了不同的分支语句。
2、进程的睡眠、同步、撤消等进程控制
用fork( )创建一个进程,再调用exec( )用新的程序替换该子进程的内容,然后利用wait( )来控制进程执行顺序。
(1)exec( )系列
系统调用exec( )系列,也可用于新程序的运行。fork( )只是将父进程的用户级上下文拷贝到新进程中,而exec( )系列可以将一个可执行的二进制文件覆盖在新进程的用户级上下文的存储空间上,以更改新进程的用户级上下文。exec( )系列中的系统调用都完成相同的功能,它们把一个新程序装入内存,来改变调用进程的执行代码,从而形成新进程。如果exec( )调用成功,调用进程将被覆盖,然后从新程序的入口开始执行,这样就产生了一个新进程,新进程的进程标识符id 与调用进程相同。
exec( )没有建立一个与调用进程并发的子进程,而是用新进程取代了原来进程。所以exec( )调用成功后,没有任何数据返回,这与fork( )不同。exec( )系列系统调用在UNIX系统库unistd.h中,共有execl、execlp、execle、execv、execvp五个,其基本功能相同,只是以不同的方式来给出参数。
一种是直接给出参数的指针,如:
int execl(path,arg0[,arg1,…argn],0);
char *path,*arg0,*arg1,…,*argn;
另一种是给出指向参数表的指针,如:
int execv(path,argv);
char *path,*argv[ ];
具体使用可参考有关书。
(2)exec( )和fork( )联合使用
系统调用exec和fork( )联合使用能为程序开发提供有力支持。用fork( )建立子进程,然后在子进程中使用exec( ),这样就实现了父进程与一个与它完全不同子进程的并发执行。
一般,wait、exec联合使用的模型为:
int status;
…
if (fork( )= =0)
{
…;
execl(…);
…;
}
wait(&status);
(3)wait( )
等待子进程运行结束。如果子进程没有完成,父进程一直等待。wait( )将调用进程挂起,直至其子进程因暂停或终止而发来软中断信号为止。如果在wait( )前已有子进程暂停或终止,则调用进程做适当处理后便返回。
系统调用格式:
int wait(status)
int *status;
其中,status是用户空间的地址。它的低8位反应子进程状态,为0表示子进程正常结束,非0则表示出现了各种各样的问题;高8位则带回了exit( )的返回值。exit( )返回值由系统给出。
核心对wait( )作以下处理:
1)首先查找调用进程是否有子进程,若无,则返回出错码;
2)若找到一处于“僵死状态”的子进程,则将子进程的执行时间加到父进程的执行时间上,并释放子进程的进程表项;
3)若未找到处于“僵死状态”的子进程,则调用进程便在可被中断的优先级上睡眠,等待其子进程发来软中断信号时被唤醒。
(4)exit( )
终止进程的执行。
系统调用格式:
void exit(status)
int status;
其中,status是返回给父进程的一个整数,以备查考。
为了及时回收进程所占用的资源并减少父进程的干预,UNIX/LINUX利用exit( )来实现进程的自我终止,通常父进程在创建子进程时,应在进程的末尾安排一条exit( ),使子进程自我终止。exit(0)表示进程正常终止,exit(1)表示进程运行有错,异常终止。
如果调用进程在执行exit( )时,其父进程正在等待它的终止,则父进程可立即得到其返回的整数。核心须为exit( )完成以下操作:
1)关闭软中断
2)回收资源
3)写记帐信息
4)置进程为“僵死状态”
3、多进程通过加锁互斥并发运行
用lockf( )来给每一个进程加锁,以实现多进程之间的互斥。
所涉及的系统调用:lockf(files,function,size),用作锁定文件的某些段或者整个文件。
本函数的头文件为
#include “unistd.h”
参数定义:
int lockf(files,function,size)
int files,function;
long size;
其中:files是文件描述符;function是锁定和解锁:1表示锁定,0表示解锁。size是锁定或解锁的字节数,为0,表示从文件的当前位置到文件尾。
4、进程间通过信号机制实现软中断通信
(1)信号的基本概念
每个信号都对应一个正整数常量(称为signal number,即信号编号。定义在系统头文件
信号与中断的相似点:
1)采用了相同的异步通信方式;
2)当检测出有信号或中断请求时,都暂停正在执行的程序而转去执行相应的处理程序;
3)都在处理完毕后返回到原来的断点;
4)对信号或中断都可进行屏蔽。
信号与中断的区别:
1)中断有优先级,而信号没有优先级,所有的信号都是平等的;
2)信号处理程序是在用户态下运行的,而中断处理程序是在核心态下运行;
(3)中断响应是及时的,而信号响应通常都有较大的时间延迟。
信号机制具有以下三方面的功能:
1)发送信号。发送信号的程序用系统调用kill( )实现;
2)预置对信号的处理方式。接收信号的程序用signal( )来实现对处理方式的预置;
3)收受信号的进程按事先的规定完成对相应事件的处理。
(2)信号的发送
信号的发送,是指由发送进程把信号送到指定进程的信号域的某一位上。如果目标进程正在一个可被中断的优先级上睡眠,核心便将它唤醒,发送进程就此结束。一个进程可能在其信号域中有多个位被置位,代表有多种类型的信号到达,但对于一类信号,进程却只能记住其中的某一个。
进程用kill( )向一个进程或一组进程发送一个信号。
(3)对信号的处理
当一个进程要进入或退出一个低优先级睡眠状态时,或一个进程即将从核心态返回用户态时,核心都要检查该进程是否已收到软中断。当进程处于核心态时,即使收到软中断也不予理睬;只有当它返回到用户态后,才处理软中断信号。对软中断信号的处理分三种情况进行:
1)如果进程收到的软中断是一个已决定要忽略的信号(function=1),进程不做任何处理便立即返回;
2)进程收到软中断后便退出(function=0);
3)执行用户设置的软中断处理程序。
(4)所涉及的中断调用
(1)kill( )
系统调用格式:int kill(pid,sig)
参数定义:int pid,sig;
其中,pid是一个或一组进程的标识符,参数sig是要发送的软中断信号。
1)pid>0时,核心将信号发送给进程pid。
2)pid=0时,核心将信号发送给与发送进程同组的所有进程。
3)pid=-1时,核心将信号发送给所有用户标识符真正等于发送进程的有效用户标识号的进程。
(2)signal( )
预置对信号的处理方式,允许调用进程控制软中断信号。
系统调用格式
signal(sig,function)
头文件为
#include
参数定义
signal(sig,function)
int sig;
void (*func) ( )
其中sig用于指定信号的类型,sig为0则表示没有收到任何信号,余者如下表:
值 名 字 说 明
01 SIGHUP 挂起(hangup)
02 SIGINT 中断,当用户从键盘按c键或break键时
03 SIGQUIT 退出,当用户从键盘按quit键时
04 SIGILL 非法指令
05 SIGTRAP 跟踪陷阱(trace trap),启动进程,跟踪代码的执行
06 SIGIOT IOT指令
07 SIGEMT EMT指令
08 SIGFPE 浮点运算溢出
09 SIGKILL 杀死、终止进程
10 SIGBUS 总线错误
11 SIGSEGV 段违例(segmentation violation),进程试图去访问其虚地址空间以外的位置
12 SIGSYS 系统调用中参数错,如系统调用号非法
13 SIGPIPE 向某个非读管道中写入数据
14 SIGALRM 闹钟。当某进程希望在某时间后接收信号时发此信号
15 SIGTERM 软件终止(software termination)
16 SIGUSR1 用户自定义信号1
17 SIGUSR2 用户自定义信号2
18 SIGCLD 某个子进程死
19 SIGPWR 电源故障
function:在该进程中的一个函数地址,在核心返回用户态时,它以软中断信号的序号作为参数调用该函数,对除了信号SIGKILL,SIGTRAP和SIGPWR以外的信号,核心自动地重新设置软中断信号处理程序的值为SIG_DFL,一个进程不能捕获SIGKILL信号。
function 的解释如下:
1)function=1时,进程对sig类信号不予理睬,亦即屏蔽了该类信号;
2)function=0时,缺省值,进程在收到sig信号后应终止自己;
3)function为非0,非1类整数时,function的值即作为信号处理程序的指针。
5、消息的发送与接收
使用系统调用msgget( ),msgsnd( ),msgrev( ),及msgctl( )编制一长度为1k的消息发送和接收的程序。
消息(message)是一个格式化的可变长的信息单元。消息机制允许由一个进程给其它任意的进程发送一个消息。当一个进程收到多个消息时,可将它们排成一个消息队列。消息使用二种重要的数据结构:一是消息首部,其中记录了一些与消息有关的信息,如消息数据的字节数;二个消息队列头表,其每一表项是作为一个消息队列的消息头,记录了消息队列的有关信息。
(1)消息机制的数据结构
(1)消息首部
记录一些与消息有关的信息,如消息的类型、大小、指向消息数据区的指针、消息队列的链接指针等。
(2)消息队列头表
其每一项作为一个消息队列的消息头,记录了消息队列的有关信息如指向消息队列中第一个消息和指向最后一个消息的指针、队列中消息的数目、队列中消息数据的总字节数、队列所允许消息数据的最大字节总数,还有最近一次执行发送操作的进程标识符和时间、最近一次执行接收操作的进程标识符和时间等。
(3) 消息队列的描述符
UNIX中,每一个消息队列都有一个称为关键字(key)的名字,是由用户指定的;消息队列有一消息队列描述符,其作用与用户文件描述符一样,也是为了方便用户和系统对消息队列的访问。
涉及的系统调用
(1) msgget( )
创建一个消息,获得一个消息的描述符。核心将搜索消息队列头表,确定是否有指定名字的消息队列。若无,核心将分配一新的消息队列头,并对它进行初始化,然后给用户返回一个消息队列描述符,否则它只是检查消息队列的许可权便返回。
系统调用格式:
msgqid=msgget(key,flag)
该函数使用头文件如下:
#include
#include
#include
参数定义
int msgget(key,flag)
key_t key;
int flag;
其中:
key是用户指定的消息队列的名字;flag是用户设置的标志和访问方式。如 IPC_CREAT |0400 是否该队列已被创建。无则创建,是则打开;
IPC_EXCL |0400 是否该队列的创建应是互斥的。
msgqid 是该系统调用返回的描述符,失败则返回-1。
(2) msgsnd()
发送一消息。向指定的消息队列发送一个消息,并将该消息链接到该消息队列的尾部。
系统调用格式:
msgsnd(msgqid,msgp,size,flag)
该函数使用头文件如下:
#include
#include
#include
参数定义:
int msgsnd(msgqid,msgp,size,flag)
I int msgqid,size,flag;
struct msgbuf * msgp;
其中msgqid是返回消息队列的描述符;msgp是指向用户消息缓冲区的一个结构体指针。缓冲区中包括消息类型和消息正文,即
{
long mtype; /消息类型/
char mtext[ ]; /消息的文本/
}
size指示由msgp指向的数据结构中字符数组的长度;即消息的长度。这个数组的最大值由MSG-MAX( )系统可调用参数来确定。flag规定当核心用尽内部缓冲空间时应执行的动作:进程是等待,还是立即返回。若在标志flag中未设置IPC_NOWAIT位,则当该消息队列中的字节数超过最大值时,或系统范围的消息数超过某一最大值时,调用msgsnd进程睡眠。若是设置IPC_NOWAIT,则在此情况下,msgsnd立即返回。
对于msgsnd( ),核心须完成以下工作:
1)对消息队列的描述符和许可权及消息长度等进行检查。若合法才继续执行,否则返回;
2)核心为消息分配消息数据区。将用户消息缓冲区中的消息正文,拷贝到消息数据区;
3)分配消息首部,并将它链入消息队列的末尾。在消息首部中须填写消息类型、消息大小和指向消息数据区的指针等数据;
4)修改消息队列头中的数据,如队列中的消息数、字节总数等。最后,唤醒等待消息的进程。
(3) msgrcv( )
接受一消息。从指定的消息队列中接收指定类型的消息。
系统调用格式:
msgrcv(msgqid,msgp,size,type,flag)
本函数使用的头文件如下:
#include
#include
#include
参数定义:
int msgrcv(msgqid,msgp,size,type,flag)
int msgqid,size,flag;
struct msgbuf *msgp;
long type;
其中,msgqid,msgp,size,flag与msgsnd中的对应参数相似,type是规定要读的消息类型,flag规定倘若该队列无消息,核心应做的操作。如此时设置了IPC_NOWAIT标志,则立即返回,若在flag中设置了MS_NOERROR,且所接收的消息大于size,则核心截断所接收的消息。
对于msgrcv系统调用,核心须完成下述工作:
1)对消息队列的描述符和许可权等进行检查。若合法,就往下执行;否则返回;
2)根据type的不同分成三种情况处理:
type=0,接收该队列的第一个消息,并将它返回给调用者;
type为正整数,接收类型type的第一个消息;
type为负整数,接收小于等于type绝对值的最低类型的第一个消息。
3)当所返回消息大小等于或小于用户的请求时,核心便将消息正文拷贝到用户区,并从消息队列中删除此消息,然后唤醒睡眠的发送进程。但如果消息长度比用户要求的大时,则做出错返回。
(4) msgctl( )
消息队列的操纵。读取消息队列的状态信息并进行修改,如查询消息队列描述符、修改它的许可权及删除该队列等。
系统调用格式:
msgctl(msgqid,cmd,buf);
本函数使用的头文件如下:
#include
#include
#include
参数定义:
int msgctl(msgqid,cmd,buf);
int msgqid,cmd;
struct msgqid_ds *buf;
其中,函数调用成功时返回0,不成功则返回-1。buf是用户缓冲区地址,供用户存放控制参数和查询结果;cmd是规定的命令。命令可分三类:
1)IPC_STAT。查询有关消息队列情况的命令。如查询队列中的消息数目、队列中的最大字节数、最后一个发送消息的进程标识符、发送时间等;
2)IPC_SET。按buf指向的结构中的值,设置和改变有关消息队列属性的命令。如改变消息队列的用户标识符、消息队列的许可权等;
3)IPC_RMID。消除消息队列的标识符。
msgqid_ds 结构定义如下:
struct msgqid_ds
{ struct ipc_perm msg_perm; /许可权结构/
short pad1[7]; /由系统使用/
ushort msg_qnum; /队列上消息数/
ushort msg_qbytes; /队列上最大字节数/
ushort msg_lspid; /最后发送消息的PID/
ushort msg_lrpid; /最后接收消息的PID/
time_t msg_stime; /最后发送消息的时间/
time_t msg_rtime; /最后接收消息的时间/
time_t msg_ctime; /最后更改时间/
};
struct ipc_perm
{ ushort uid; /当前用户/
ushort gid; /当前进程组/
ushort cuid; /创建用户/
ushort cgid; /创建进程组/
ushort mode; /存取许可权/
{ short pid1; long pad2;} /由系统使用/
}
6、进程的共享存储区通信
编制一长度为1k的共享存储区发送和接收的程序。
(1)共享存储区机制的概念
共享存储区(Share Memory)是UNIX系统中通信速度最高的一种通信机制。该机制可使若干进程共享主存中的某一个区域,且使该区域出现(映射)在多个进程的虚地址空间中。另一方面,一个进程的虚地址空间中又可连接多个共享存储区,每个共享存储区都有自己的名字。当进程间欲利用共享存储区进行通信时,必须先在主存中建立一共享存储区,然后将它附接到自己的虚地址空间上。此后,进程对该区的访问操作,与对其虚地址空间的其它部分的操作完全相同。进程之间便可通过对共享存储区中数据的读、写来进行直接通信。图示列出二个进程通过共享一个共享存储区来进行通信的例子。其中,进程A将建立的共享存储区附接到自己的AA’区域,进程B将它附接到自己的BB’区域。
进程A的虚空间 内存空间 进程B的虚空间
A
A’
应当指出,共享存储区机制只为进程提供了用于实现通信的共享存储区和对共享存储区进行操作的手段,然而并未提供对该区进行互斥访问及进程同步的措施。因而当用户需要使用该机制时,必须自己设置同步和互斥措施才能保证实现正确的通信。
(2)涉及的系统调用
1)shmget( )
创建、获得一个共享存储区。
系统调用格式:
shmid=shmget(key,size,flag)
该函数使用头文件如下:
#include
#include
#include
参数定义
int shmget(key,size,flag);
key_t key;
int size,flag;
其中,key是共享存储区的名字;size是其大小(以字节计);flag是用户设置的标志,如IPC_CREAT。IPC_CREAT表示若系统中尚无指名的共享存储区,则由核心建立一个共享存储区;若系统中已有共享存储区,便忽略IPC_CREAT。
附:
操作允许权 八进制数
用户可读 00400
用户可写 00200
小组可读 00040
小组可写 00020
其它可读 00004
其它可写 00002
控制命令 值
IPC_CREAT 0001000
IPC_EXCL 0002000
例:shmid=shmget(key,size,(IPC_CREAT|0400))
创建一个关键字为key,长度为size的共享存储区
2)shmat( )
共享存储区的附接。从逻辑上将一个共享存储区附接到进程的虚拟地址空间上。
系统调用格式:
virtaddr=shmat(shmid,addr,flag)
该函数使用头文件如下:
#include
#include
#include
参数定义
char *shmat(shmid,addr,flag);
int shmid,flag;
char * addr;
其中,shmid是共享存储区的标识符;addr是用户给定的,将共享存储区附接到进程的虚地址空间;flag规定共享存储区的读、写权限,以及系统是否应对用户规定的地址做舍入操作。其值为SHM_RDONLY时,表示只能读;其值为0时,表示可读、可写;其值为SHM_RND(取整)时,表示操作系统在必要时舍去这个地址。该系统调用的返回值是共享存储区所附接到的进程虚地址viraddr。
3)shmdt( )
把一个共享存储区从指定进程的虚地址空间断开。
系统调用格式:
shmdt(addr)
该函数使用头文件如下:
#include
#include
#include
参数定义
int shmdt(addr);
char addr;
其中,addr是要断开连接的虚地址,亦即以前由连接的系统调用shmat( )所返回的虚地址。调用成功时,返回0值,调用不成功,返回-1。
4)shmctl( )
共享存储区的控制,对其状态信息进行读取和修改。
系统调用格式:
shmctl(shmid,cmd,buf)
该函数使用头文件如下:
#include
#include
#include
参数定义
int shmctl(shmid,cmd,buf);
int shmid,cmd;
struct shmid_ds *buf;
其中,buf是用户缓冲区地址,cmd是操作命令。命令可分为多种类型:
第一种:用于查询有关共享存储区的情况。如其长度、当前连接的进程数、共享区的创建者标识符等;
第二种:用于设置或改变共享存储区的属性。如共享存储区的许可权、当前连接的进程计数等;
第三种:对共享存储区的加锁和解锁命令;
第四种:删除共享存储区标识符等。
上述的查询是将shmid所指示的数据结构中的有关成员,放入所指示的缓冲区中;而设置是用由buf所指示的缓冲区内容来设置由shmid所指示的数据结构中的相应成员。
四、实验中用到的系统调用函数(包括实验原理中介绍的和自己采用的),自己采用的系统调用函数要按照指导书中的格式说明进行介绍。
本实验的用到的主要系统调用函数:
Fork, exec, wait, exit, getpid, sleep, lockf, kill, signal, read, write, msgget, msgsnd, msgrcv, msgctl,shmget, shmat, shmdt, shmctl。
五、实验步骤(要求写出实验过程和思路。)
1.父进程创建两个子进程,然后三个进程并发执行,父进程输出a,子进程a输出b,子进程b输出c,多次运行程序,并通过结果观察其规律。
2.父进程创建一个子进程,如果子进程创建成功,若此时是父进程正在运行,则父进程进行等待操作,将cpu教给子进程,子进程获取cpu后,开始运行,用系统中bin目录下的ls命令程序装入子进程运行的地址,即用ls命令程序代替子进程。当子进程运行完毕,父进程输出,程序运行完毕。
3.父进程创建两个子进程,若是父进程运行,则输出parent 1-500,若子进程运行,则输出son1-500或者daughter1-500,分别用加锁和不加锁的程序进行测试。
4.用fork( )创建两个子进程,再用系统调用signal( )让父进程捕捉键盘上来的中断信号(即按^c键);捕捉到中断信号后,父进程用系统调用kill( )向两个子进程发出信号,子进程捕捉到信号后分别输出下列信息后终止:
Child process1 is killed by parent!
Child process2 is killed by parent!
父进程等待两个子进程终止后,输出如下的信息后终止:
Parent process is killed!
5.在两个终端上创建两个进程,分别是client负责发送消息,server负责接受消息。
6.建立两个进程server和client,开辟一个共享资源区,双方都处于不断同步的状态,client等待资源区被server修改,而server修改后,也在等待client修改资源区,client察觉资源区被修改,也会继续修改,就这样进行循环。
六、实验数据及源代码(学生必须提交自己设计的程序源代码,并有注释,源代码电子版也一并提交),包括思考题的程序。
1)
#include
#include
#include
int main(int argc,char *argv[])
{
int pa; //进程a
int pb; //进程b
while((pa=fork())==-1); //等待子进程a建立完成
if(pa==0)
{
putchar('b');
}
else
{
while((pb=fork())==-1);
if(pb==0)
{
putchar('c');
}
else
{
putchar('a');
}
}
}
2)
#include
#include
#include
#include
int main(int argc,char* argv[])
{
int pid;
pid=fork();
switch(pid)
{
case -1:
printf("fork fail!\n");
exit(1);
case 0:
execl("/bin/ls","-1","-color",NULL);
printf("exec fail!\n");
exit(1);
default:
wait(NULL);
printf("ls completed!\n");
exit(0);
}
}
3)
#include
#include
#include
#include
int main(int argc,char* argv[])
{
int p1,p2,i;
p1=fork();
p2=fork();
if(p1==0)
{
lockf(1,1,0); //加锁
for(i=0;i<500;i++)
printf("parent %d\n",i);
lockf(1,0,0); //解锁
wait(0); //保证子进程终止前,父进程不终止
exit(0);
}
else
{
if(p2==0)
{
lockf(1,1,0);
for(i=0;i<500;i++)
printf("son %d\n",i);
lockf(1,0,0);
wait(0);
exit(0);
}
else
{
lockf(1,1,0);
for(i=0;i<500;i++)
printf("daughter %d\n",i);
lockf(1,0,0);
exit(0);
}
}
}
4)
#include
#include
#include
#include
#include
int pid1,pid2;
int EndFlag=0;
int pf1=0;
int pf2=0;
void IntDelete()
{
kill(pid1,16);
kill(pid2,17);
}
void Int1()
{
printf("child process 1 is killed by parent!\n");
exit(0);
}
void Int2()
{
printf("child process 2 is killed by parent!\n");
exit(0);
}
int main(int argc,char*argv[])
{
int exitpid;
if(pid1=fork())
{
if(pid2=fork())
{
signal(SIGINT,IntDelete);
waitpid(-1,&exitpid,0);
waitpid(-1,&exitpid,0);
printf("parent process is killed\n");
exit(0);
}
else
{
signal(SIGINT,SIG_IGN);
signal(17,Int2);
pause();
}
}
else
{
signal(SIGINT,SIG_IGN);
signal(16,Int1);
pause();
}
}
5)
Server:
#include
#include
#include
#include
#include
#define MSGKEY 75
struct msgform
{
long mtype;
char mtextp[1000];
}msg;
int msgqid;
void server()
{
msgqid=msgget(MSGKEY,0777|IPC_CREAT);
do
{
msgrcv(msgqid,&msg,1030,0,0);
printf("(sever)received\n");
}while(msg.mtype!=1);
msgctl(msgqid,IPC_RMID,0);
exit(0);
}
int main(int argc,char* argv[])
{
server();
}
client:
#include
#include
#include
#include
#include
#define MSGKEY 75
struct magform
{
long mtype;
char mtext[1000];
}msg;
int msgqid;
void client()
{
int i;
msgqid=msgget(MSGKEY,0777);
for(i=10;i>=-1;i--)
{
msg.mtype=i;
printf("(client)sent\n");
msgsnd(msgqid,&msg,1024,0);
}
exit(0);
}
int main(int argc,char* argv[])
{
client();
}
~
6)
#include
#include
#include
#include
#include
#include
#include
#define SHMKEY 75
int shmid,i;
int *addr;
void client()
{
int i;
shmid=shmget(SHMKEY,1024,0777);
addr=shmat(shmid,0,0);
for(i=9;i>=0;i--)
{
while(*addr!=-1)
printf("(client)sent\n");
*addr=i;
}
exit(0);
}
void server()
{
shmid=shmget(SHMKEY,1024,0777|IPC_CREAT);
addr=shmat(shmid,0,0);
do
{
*addr=-1;
while(*addr==-1);
printf("*(server)receive\n");
}while(*addr);
shmctl(shmid,IPC_RMID,0);
exit(0);
}
int main(int argc,char*argv[])
{
while((i=fork())==-1);
if(!i) server();
system("ipcs -m");
while((i=fork())==-1);
if(!i) client();
wait(0);
wait(0);
}
七、实验结果分析(截屏的实验结果,与实验结果对应的实验分析)
1、实验结果与实验程序、实验步骤、实验原理、操作系统原理的对应分析;
2、不同条件下的实验结果反应的问题及原因;
3、实验结果的算法时间、效率、鲁棒性等性能分析。
结果:
刚开始的结果都是acb。
但是这不太符合并发执行的进程结果,所以后来我把虚拟机的处理器内核数量从1提高到3就会出现随机的结果啦!
分析:从结果来看,父进程和两个子进程的并发执行,结果是随机的,即abc的顺序都有可能,也就是说,并发执行的子进程之间,在没有优先级的限制条件下,争夺cpu的情况是随机的,在代码中,可能存在进程建立时间和代码运行输出等时间的一个不对等性,但是进程获取的时间片是随机的,故而会输出随机的不同结果。
思考题:
2.
结果:
利用wait()函数实现同步:
未用wait()函数实现同步:
分析:第一个结果是利用睡眠等待等操作实现同步,第二幅图的结果是未用等待操作导致不同步的结果,由第一幅图可知,子进程创建成功,且运行了ls命令程序,多次运行程序也是同一个结果,而当取消wait()的时候,顺序输出便错乱了,说明该程序父进程和子进程实现了同步。
3.
结果:
1)不用lockf()进行进程加锁:
利用lockf()给进程加锁:
分析:
Lockf(1,1,0)加锁输出设备,而Lockf(1,0,0)解锁输出设备,未用lockf进行进程加锁,三个循环之间会交错进行,输出也会出现交错状态,但加锁后则不会,所以在lockf(1,1,0)和lockf(1,0,0)之间的for循环输出中,不会被间断。但加锁前的三个进程间的存在并发执行,故而parent,daughter,son这三个循环的输出顺序随机,但循环不会被间断。
分析:
结果:
分析:
父进程接收到ctrl+c信号才会调用kill()向两个子进程发出信号。子进程接收到信号才会打印。
5)
结果:
分析:首先运行服务代码test5_s,server会进入等待状态,继而运行客户代码test5_c像服务器发出消息,并输出(client)sent,继而运行服务器的终端收到消息,输出(server)received。
6)
结果:
分析:
Client和server两个进程拥有共享资源区,开始时,client修改了资源区内容,然后等待共享区的内容被修改为-1,因此是等待状态,而server修改了共享区的内容后,client察觉到内容变为-1,故而又继续修改,不断循环,直到client修改为-1。
1、进程创建与进程并发执行
(1)系统是怎样创建进程的?
1.申请空白PCB(进程控制块);
2.为新进程分配资源;
3.初始化PCB;
4.就新进程插入就绪队列;
(2)当首次调用新创建进程时,其入口在哪里?
fork()函数被调用一次,但返回两次;两次返回区别在于:子程序返回值是0,而父进程返回值是子进程的ID。子进程和父进程运行相同的代码,但是有自己的数据空间。
(3)程序的多次运行结果为什么不同?如何控制实验结果的随机性?
多个进程的并发执行,每个进程都有先获取cpu的可能性,故而哪一个进程先执行是随机的。可以通过等待,睡眠的等操作来实现多个进程的同步。
(4)利用strace 和ltrace -f -i -S ./executable-file-name查看程序执行过程,并分析原因,画出进程家族树。
2、进程的睡眠、同步、撤消等进程控制
(1)可执行文件加载时进行了哪些处理?
进程用exec( )装入命令ls ,exec( )后,子进程的代码被ls的代码取代,这时子进程的PC指向ls的第1条语句,开始执行ls的命令代码
(2)什么是进程同步?wait( )是如何实现进程同步的?
进程同步是指多个相关进程在执行次序上进行协调,以使并发执行的主进程之间有效的共享资源和相互合作,从而使程序的执行具有可再现性。
首先程序在调用fork()创建了一个子进程后,马上调用wait(),使父进程在子进程调用之前一直处于睡眠状态,这样使子进程先运行,子进程运行exec()装入命令后,然后调用wait(0),使子进程和父进程并发执行,实现了进程同步。
(3)wait( )和exit()是如何控制实验结果的随机性的?
可以看出在使用了exec()函数后程序使用了ls的命令,列出/bin/目录下的文件信息,执行完execl()函数后,子进程调用exit()函数,退出当前进程,我们可以发现在使用wait()函数后,父进程永远将在其他的子进程完成之后才执行,所以在输出的结果中我们可以看到最后输出的将是父进程的信息,这样进而可以控制实验结果的随机性。
3、多进程通过加锁互斥并发运行
(1)进程加锁和未上锁的输出结果相同吗? 为什么?
大致与未上锁的输出结果相同,也是随着执行时间不同,输出结果的顺序有所不同。
不同进程之间不存在共享临界资源问题,所以加锁与不加锁的效果大致相同。
4、进程间通过信号机制实现软中断通信
(1)为了得到实验内容要求的结果,需要用到哪些系统调用函数来实现及进程间的通信控制和同步?
Signal,kill,wait,exit,lockf
(1)kill( )和signal( )函数在信号通信中的作用是什么?如果分别注释掉它们,结果会如何?
结果都不会有输出,因为注释kill的情况下,父进程在等待子进程的信号,子进程也在等待父进程的信号,进入的死锁状态。而注释掉signal的情况下,子进程无法接受道父进程的信号,也无法向父进程发送信号,所以子进程结束后,父进程一直处于等待状态。
5、消息的发送与接收
(1)为了便于操作和观察结果,需要编制几个程序分别用于消息的发送与接收?
为了便于操作和观察结果,编制了两个程序client.c和server.c,分别用于信息的发送与接受。
(2)这些程序如何进行编辑、编译和执行?为什么?
使用vim工具:vim test5_c.c vim test5_s.c分别编辑两个文件的代码
使用gcc: gcc -o test5_c test5_c.c和gcc -o test5_s test5_s.c
最后在两个终端分别先后运行./test5_s和./test5_c
(3)如何实现消息的发送与接收的同步?
分别在两个终端建立两个进程server和client,client发送信号,server接受信号,先运行server,后运行client。在server接受到消息后才会输出。
6、进程的共享存储区通信
(1)为了便于操作和观察结果,需要如何合理设计程序来实现子进程间的共享存储区通信?
建立两个进程server和client,开辟一个共享资源区,双方都处于不断同步的状态,client等待资源区被server修改,而server修改后,也在等待client修改资源区,client察觉资源区被修改,也会继续修改,就这样进行循环多次。
(2)比较消息通信和共享存储区通信这两种进程通信机制的性能和优缺点。
答:由于两种机制实现的机理和用处都不一样,难以直接进行时间上的比较。如果比较其性能,应更加全面的分析。
(1)消息队列的建立比共享区的设立消耗的资源少。前者只是一个软件上设定的问题,后者需要对硬件的操作,实现内存的映像,当然控制起来比前者复杂。如果每次都重新进行队列或共享的建立,共享区的设立没有什么优势。
(2)当消息队列和共享区建立好后,共享区的数据传输,受到了系统硬件的支持,不耗费多余的资源;而消息传递,由软件进行控制和实现,需要消耗一定的cpu的资源。从这个意义上讲,共享区更适合频繁和大量的数据传输。
(3)消息的传递,自身就带有同步的控制。当等到消息的时候,进程进入睡眠状态,不再消耗cpu资源。而共享队列如果不借助其他机制进行同步,接收数据的一方必须进行不断的查询,白白浪费了大量的cpu资源。可见,消息方式的使用更加灵活。