Download presentation
Presentation is loading. Please wait.
1
第 7 章 进程间的通信
2
本章重点 进程通信中信号概念及信号处理 进程间的管道通信编程 进程间的内存共享编程
3
7.1.1 信号及其使用 信号是在软件层次上对中断机制的一种模拟,是一种异步通信方式。信号可以直接进行用户空间进程和内核进程之间的交互,内核进程也可以利用它来通知用户空间进程发生了哪些系统事件。 信号事件的发生有两个来源: 硬件来源,如按下了键盘Delete键或者鼠标单击,通常产生中断信号(SIGINT)或者其它硬件故障。 软件来源,如使用系统调用或者是命令发出信号。最常用发送信号的系统函数是kill、raise、alarm、setitimer、sigation和sigqueue函数,软件来源还包括一些非法运算等操作。
4
7.1.1 信号及其使用 例7.1:列出系统所支持的所有信号列表。
root]#kill -l (2)分析: SIG信号 SIGRTMIN信号是从UNIX系统中继承下来的称为不可靠信号(也称为非实时信号)。 SIGRTMAX是为了解决前面“不可靠信号”问题而进行更改和扩充的信号,称为可靠信号(也称为实时信号)。 可靠信号(实时信号):支持排队,发送用户进程一次就注册一次,发现相同信号已经在进程中注册,也要再注册。 不可靠信号(非实时信号):不支持排队,发送用户进程判断后注册,发现相同信号已经在进程中注册,就不再注册,忽略该信号。前面显示的31种“SIG”开头的,也属于非实时信号。
5
7.1.1 信号及其使用 一旦有信号产生,用户进程对信号的响应有3种方式: 执行默认操作。Linux对每种信号都规定了默认操作。
捕捉信号。定义信号处理函数,当信号发生时,执行相应的处理函数。 忽略信号。不希望接收到的信号对进程的执行产生影响,而让进程继续进行时,可以忽略该信号,即不对信号进程任何处理。
6
常见信号的含义及其默认操作
7
7.2.1 信号操作的相关函数
8
7.2.1 信号操作的相关函数 1. 信号发送 信号发送的关键,是使系统知道向哪个进程发送以及发送什么信号。能否向某一进程发送某一特定信号是和用户的权限密切相关的。
9
7.2.1 信号操作的相关函数 例7.2:设计一个程序,要求用户进程复制出一个子进程,父进程向子进程发出信号,子进程收到此信号,结束子进程。
源程序代码: 编译成功后,运行可执行文件,此时系统会显示子进程的进程号(PID)、kill 函数的返回值和SIGKILL信号所结束进程的进程号(PID)。 由此例可知,系统调用kill函数和raise函数,都是简单地向某一进程发送信号。kill函数用于给特定的进程或进程组发送信号,raise函数用于向一个进程自身发送信号。
10
7.2.1 信号操作的相关函数
11
7.2.1 信号操作的相关函数 2. 信号处理 当某个信号被发送到一个正在运行的进程时,该进程即对此特定信号注册相应的信号处理函数,以完成所需处理。
12
7.2.1 信号操作的相关函数 例7.3:设计一个程序,要求程序运行后进入无限循环,当用户按下中断键(Ctrl+C)时,进入程序的自定义信号处理函数,当用户再次按下中断键(Ctrl+C)后,结束程序运行。 源程序代码 : signal函数主要用于前31种非实时信号的处理,不支持信号传递信息(函数类型是void),但使用简单、方便,只需把要处理的信号和处理函数列出即可,因此受到许多软件工程师欢迎。
13
7.2.1 信号操作的相关函数
14
7.2.1 信号操作的相关函数 3. 信号阻塞 有时既不希望进程在接收到信号时立刻中断进程的执行,也不希望此信号完全被忽略掉,而是延迟一段时间再去调用信号处理函数,这个时候就需要信号阻塞来完成。
15
7.2.1 信号操作的相关函数 例7.4:设计一个程序,要求程序主体运行时,即使用户按下的中断键(Ctrl+C),也不能影响正在运行的程序,等程序主体运行完毕后才进入自定义信号处理函数。 源程序代码 :
16
7.2.1 信号操作的相关函数
17
7.2 管道 在Linux中,管道是一种特殊的文件,对一个进程来说,管道的写入和读取与一个普通文件没有区别。
管道是Linux支持的最初Unix IPC形式之一,具有以下特点: 管道是半双工的,数据只能向一个方向流动;需要双方通信时,需要建立起两个管道; 只能用于父子进程或者兄弟进程之间(具有亲缘关系的进程); 单独构成一种独立的文件系统:管道对于管道两端的进程而言,就是一个文件,但它不是普通的文件,它不属于某种文件系统,而是自立门户,单独构成一种文件系统,并且只存在与内存中。
18
7.2.1 低级管道操作 低级管道操作时,建立管道用pipe函数,建立管道后Linux系统会同时为该进程建立2个文件描述符pipe_fd[0]和pipe_fd[1]。pipe_fd[0]用来从管道读取数据,pipe_fd[1]用来把数据写入管道。
19
7.2.1 低级管道操作 例7.5:设计一个程序,要求创建一个管道,复制进程,父进程往管道中写入字符串,子进程从管道中读取前输出字符串。
源程序代码:
20
7.2.1 低级管道操作
21
7.2.2 高级管道操作 例7.6:设计一个程序,要求用popen创建管道,实现“ls -l|grep 7-6”的功能。 源程序代码 :
使用popen函数读写管道,实际上也是调用pipe函数建立一个管道,再调用fork函数建立子进程,接着会建立一个shell环境,并在这个shell环境中执行参数指定的进程。
22
7.2.2 高级管道操作
23
7.2.3 命名管道 若要在两个不相关的进程之间用管道通信,需要用到命名管道FIFO。
命名管道FIFO是通过Linux系统中的文件进行通信。命名管道的创建一般用mkfifo函数,创建成功后,就使用open、read、write等函数传输数据。
24
7.2.3 命名管道 例7.7:设计两个程序,要求用命名管道FIFO,实现简单的聊天功能。
25
7.2.3 命名管道 源程序7-7zhang.c代码:
26
7.2.3 命名管道 7-7li.c程序代码如下:
27
7.2.3 命名管道 mkfifo函数说明 memset函数说明
28
7.3 消息队列 消息队列,就是一个消息的链表,是一系列保存在内核中的消息的列表。用
消息队列的优势:对每个消息指定特定消息类型,接收的时候不需要按队列次序,而是可以根据自定义条件接收特定类型的消息。 可以把消息看作一个记录,具有特定的格式以及特定的优先级。对消息队列有写权限的进程可以按照一定的规则添加新消息;对消息队列有读权限的进程则可以从消息队列中读取消息。 消息队列的常用函数
29
7.3 消息队列 例7.8:设计一个程序,要求创建消息队列,输入的文字添加到消息队列后,读取队列中的消息输出。 源程序代码:
由此例可知,进程间通过消息队列通信,主要是创建或打开消息队列、添加消息、读取消息和控制消息队列这四种操作。
30
ftok函数说明 msgget函数说明 msgsnd函数说明 msgrcv函数说明
31
7.4 共享内存 共享内存允许两个或多个进程共享一给定的存储区,因为数据不需要来回复制,所以是最快的一种进程间通信机制。 共享内存原理:
32
7.4 共享内存 共享内存可以通过mmap()系统调用(特殊情况下还可以采用匿名映射)机制实现,也可以通过系统V共享内存机制实现。应用接口和原理很简单,内部机制复杂。为了实现更安全通信,往往还与信号灯等同步机制共同使用。 常用函数 :
33
7.4.1 mmap系统调用 例7.9:设计一个程序,要求复制进程,父子进程通过匿名映射实现共享内存。 源程序代码:
34
7.4.1 mmap系统调用 mmap函数说明 munmap函数说明
35
7.4.2 系统V共享内存 系统V共享内存指的是把所有共享数据放在共享内存区域(IPC shared memory region),任何想要访问该数据的进程都必须在本进程的地址空间新增一块内存区域,用来映射存放共享数据的物理内存页面。 系统V共享内存是通过映射特殊文件系统shm中的文件实现进程间的共享内存通信。
36
7.4.2 系统V共享内存 例7.10:设计两个程序,要求通过系统V共享内存通信,一个程序写入系统V共享区域,另一个程序读取系统V共享区域。
7-10write.c程序代码:
37
7.4.2 系统V共享内存 7-10read.c程序代码:
38
7.4.2 系统V共享内存 结论: 系统V共享内存中的数据,从来不写入到实际磁盘文件中去;而通过mmap()映射普通文件实现的共享内存通信可以指定何时将数据写入磁盘文件中。 系统V共享内存是随内核持续的,即使所有访问共享内存的进程都已经正常终止,共享内存区仍然存在(除非显式删除共享内存),在内核重新引导之前,对该共享内存区域的任何改写操作都将一直保留。 通过调用mmap()映射普通文件进行进程间通信时,一定要注意考虑进程何时终止对通信的影响。而通过系统V共享内存实现通信的进程则不然。
39
7.4.2 系统V共享内存 shmget函数说明 shmat函数说明 shmdt函数说明
40
思考与实验 设计一个程序,要求程序运行后进入一个无限循环,当用户按下中断键(Ctrl+Z)时,进入程序的自定义信号处理函数,当用户再次按下中断键(Ctrl+Z)后,结束程序运行。 设计一个程序,要求程序主体运行时,即使用户按下的中断键(Ctrl+C),也不能影响正在运行的程序,等程序主体运行完毕后才进入自定义信号处理函数。 设计一个程序,要求创建一个管道PIPE,复制进程,父进程运行命令“ls -l”,把运行结果写入管道,子进程从管道中读取“ls -l”的结果,把读出的作为输入接着运行“grep 7-5”。
41
思考与实验 设计两个程序,要求用命名管道FIFO,实现简单的文本文件或图片文件的传输功能。
设计两个程序,要求用消息队列,实现聊天程序,每次发言后自动在后面增加当前系统时间。增加结束字符,比如最后输入“88”后结束进程。 设计两个程序,要求用mmap系统,实现简单的聊天程序。
Similar presentations