线程同步——信号量

上传人:桔**** 文档编号:473655379 上传时间:2023-04-22 格式:DOC 页数:16 大小:53KB
返回 下载 相关 举报
线程同步——信号量_第1页
第1页 / 共16页
线程同步——信号量_第2页
第2页 / 共16页
线程同步——信号量_第3页
第3页 / 共16页
线程同步——信号量_第4页
第4页 / 共16页
线程同步——信号量_第5页
第5页 / 共16页
点击查看更多>>
资源描述

《线程同步——信号量》由会员分享,可在线阅读,更多相关《线程同步——信号量(16页珍藏版)》请在金锄头文库上搜索。

1、信号是E.W.Dijkstra在二十世纪六十年代末设计的一种编程架构。Dijkstra的模型与铁路操作有关:假设某段铁路是单线的,因此一次只允许一列火车通过。信号将用于同步通过该轨道的火车。火车在进入单一轨道之前必须等待信号灯变为允许通行的状态。火车进入轨道后,会改变信号状态,防止其他火车进入该轨道。火车离开这段轨道时,必须再次更改信号的状态,以便允许其他火车进入轨道。在计算机版本中,信号以简单整数来表示。线程等待获得许可以便继续运行,然后发出信号,表示该线程已经通过针对信号执行操作来继续运行。线程必须等到信号的值为正,然后才能通过将信号值减1来更改该值。完成此操作后,线程会执行操作,即通过将

2、信号值加1来更改该值。这些操作必须以原子方式执行,不能再将其划分成子操作,即,在这些子操作之间不能对信号执行其他操作。在操作中,信号值在减小之前必须为正,从而确保生成的信号值不为负,并且比该值减小之前小1。在和操作中,必须在没有干扰的情况下进行运算。如果针对同一信号同时执行两个操作,则实际结果是信号的新值比原来大2。对于大多数人来说,如同记住Dijkstra是荷兰人一样,记住和本身的含义并不重要。但是,从真正学术的角度来说,代表prolagen,这是由proberenteverlagen演变而来的杜撰词,其意思是尝试减小。代表verhogen,其意思是增加。Dijkstra的技术说明EWD74

3、中介绍了这些含义。(3RT)和(3RT)分别与Dijkstra的和操作相对应。(3RT)是操作的一种条件形式。如果调用线程不等待就不能减小信号的值,则该调用会立即返回一个非零值。有两种基本信号:二进制信号和计数信号量。二进制信号的值只能是0或1,计数信号量可以是任意非负值。二进制信号在逻辑上相当于一个互斥锁。不过,尽管不会强制,但互斥锁应当仅由持有该锁的线程来解除锁定。因为不存在“持有信号的线程”这一概念,所以,任何线程都可以执行或(3RT)操作。计数信号量与互斥锁一起使用时的功能几乎与条件变量一样强大。在许多情况下,使用计数信号量实现的代码比使用条件变量实现的代码更为简单。但是,将互斥锁用于

4、条件变量时,会存在一个隐含的括号。该括号可以清楚表明程序受保护的部分。对于信号则不必如此,可以使用并发编程当中的goto对其进行调用。信号的功能强大,但是容易以非结构化的不确定方式使用。1、命名信号量和未命名信号量POSIX信号可以是未命名的,也可以是命名的。未命名信号在进程内存中分配,并会进行初始化。未命名信号可能可供多个进程使用,具体取决于信号的分配和初始化的方式。未命名信号可以是通过fork()继承的专用信号,也可以通过用来分配和映射这些信号的常规文件的访问保护功能对其进行保护。命名信号类似于进程共享的信号,区别在于命名信号是使用路径名而非pshared值引用的。命名信号可以由多个进程共

5、享。命名信号具有属主用户ID、组ID和保护模式。对于open、retrieve、close和remove命名信号,可以使用以下函数:sem_open、sem_getvalue、sem_close和sem_unlink。通过使用sem_open,可以仓U建一个命名信号,其名称是在文件系统的名称空间中定义的。2、计数信号量概述从概念上来说,信号量是一个非负整数计数。信号量通常用来协调对资源的访问,其中信号计数会初始化为可用资源的数目。然后,线程在资源增加时会增加计数,在删除资源时会减小计数,这些操作都以原子方式执行。如果信号计数变为零,则表明已无可用资源。计数为零时,尝试减小信号的线程会被阻塞,直

6、到计数大于零为止。由于信号无需由同一个线程来获取和释放,因此信号可用于异步事件通知,如用于信号处理程序中。同时,由于信号包含状态,因此可以异步方式使用,而不用象条件变量那样要求获取互斥锁。但是,信号的效率不如互斥锁高。缺省情况下,如果有多个线程正在等待信号,则解除阻塞的顺序是不确定的。信号在使用前必须先初始化,但是信号没有属性。3、初始化信号量使用sem_init(3RT)可以将sem所指示的未命名信号变量初始化为value。sem_init语法intsem_init(sem_t*sem,intpshared,unsignedintvalue);#includesem_tsem;intpsha

7、red;intret;intvalue;pshared=0;value=1;ret=sem_init(&sem,pshared,value);如果pshared的值为零,则不能在进程之间共享信号。如果pshared的值不为零,则可以在进程之间共享信号。(1) 多个线程决不能初始化同一个信号。(2) 不得对其他线程正在使用的信号重新初始化4、初始化进程内信号量pshared为0时,信号只能由该进程内的所有线程使用。#includesem_tsem;intret;intcount=4;ret=sem_init(&sem,0,count);5、初始化进程间信号量pshared不为零时,信号可以由其他

8、进程共享#includesem_tsem;intret;intcount=4;ret=sem_init(&sem,1,count);6、sem_init返回值sem_init()在成功完成之后会返回零。其他任何返回值都表示出现了错误。如果出现以下任一情况,该函数将失败并返回对应的值。EINVAL描述:参数值超过了SEM_VALUE_MAX。ENOSPC描述:初始化信号所需的资源已经用完。到达信号的SEM_NSEMS_MAX限制。ENOSYS描述:系统不支持sem_init()函数。EPERM描述:进程缺少初始化信号所需的适当权限。7、增加信号sem_post语法intsem_post(sem_

9、t*sem);#includesem_tsem;intret;ret=sem_post(&sem);如果所有线程均基于信号阻塞,则会对其中一个线程解除阻塞。sem_post返回值sem_post()在成功完成之后会返回零。其他任何返回值都表示出现了错误。如果出现以下情况,该函数将失败并返回对应的值。EINVAL描述:sem所指示的地址非法。8、基于信号计数进行阻塞使用sem_wait(3RT)可以阻塞调用线程,直到sem所指示的信号计数大于零为止,之后以原子方式减小计数。sem_wait语法intsem_wait(sem_t*sem);#includesem_tsem;intret;ret=s

10、em_wait(&sem);sem_wait返回值sem_wait()在成功完成之后会返回零。其他任何返回值都表示出现了错误。如果出现以下任一情况,该函数将失败并返回对应的值。EINVAL描述:sem所指示的地址非法。EINTR描述:此函数已被信号中断。9、减小信号计数使用sem_trywait(3RT)可以在计数大于零时,尝试以原子方式减小sem所指示的信号计数。sem_trywait语法intsem_trywait(sem_t*sem);#includesem_tsem;intret;ret=sem_trywait(&sem);此函数是sem_wait()的非阻塞版本。sem_trywai

11、t()在失败时会立即返回。sem_trywait返回值sem_trywait()在成功完成之后会返回零。其他任何返回值都表示出现了错误。如果出现以下任一情况,该函数将失败并返回对应的值。EINVAL描述:sem所指示的地址非法。EINTR描述:此函数已被信号中断。EAGAIN描述:信号已为锁定状态,因此该信号不能通过sem_trywait()操作立即锁定。10、销毁信号状态使用sem_destroy(3RT)可以销毁与sem所指示的未命名信号相关联的任何状态。sem_destroy语法intsem_destroy(sem_t*sem);#includesem_tsem;intret;ret=s

12、em_destroy(&sem);sem_destroy返回值sem_destroy()在成功完成之后会返回零。其他任何返回值都表示出现了错误。如果出现以下情况,该函数将失败并返回对应的值。EINVAL描述:sem所指示的地址非法。例子1:在这个例子中,一共有4个线程,其中两个线程负责从文件读取数据到公共的缓冲区,另两个线程从缓冲区读取数据作不同的处理(加和乘运算)。#include#include#include#defineMAXSTACK100intstackMAXSTACK2;intsize=0;sem_tsem;从文件l.dat读取数据,每读一次,信号量加1voidReadData1

13、(void)FILE*fp=fopen(1.dat,r);while(!feof(fp)fscanf(fp,%d%d,&stacksize0,&stacksize1);sem_post(&sem);+size;fclose(fp);从文件2.dat读取数据,每读一次,信号量减1voidReadData2(void)FILE*fp=fopen(2.dat,r);while(!feof(fp)fscanf(fp,%d%d,&stacksize0,&stacksize1);sem_post(&sem);+size;fclose(fp);/阻塞等待缓冲区有数据,读取数据进行加法运算后,释放空间,继续等

14、待voidHandleData1(void)while(1)sem_wait(&sem);printf(Plus:%d+%d=%dn,stacksize0,stacksize1,stacksize0+stacksize1);-size;/阻塞等待缓冲区有数据,读取数据进行乘法运算后,释放空间,继续等待voidHandleData2(void)while(1)sem_wait(&sem);printf(Multiply:%d*%d=%dn,stacksize0,stacksize1,stacksize0*stacksize1);-size;intmain(void)pthread_tt1,t2,

15、t3,t4;sem_init(&sem,0,0);pthread_create(&t1,NULL,(void*)HandleData1,NULL);pthread_create(&t2,NULL,(void*)HandleData2,NULL);pthread_create(&t3,NULL,(void*)ReadData1,NULL);pthread_create(&t4,NULL,(void*)ReadData2,NULL);pthread_join(t1,NULL);在Linux下,我们用命令gcc-lpthreadsem.c-osem生成可执行文件sem。我们事先编辑好数据文件1.dat和2.dat,假设它们的内容分另U为12345678910和-1-2-3-4-5-6

展开阅读全文
相关资源
正为您匹配相似的精品文档
相关搜索

最新文档


当前位置:首页 > 办公文档 > 解决方案

电脑版 |金锄头文库版权所有
经营许可证:蜀ICP备13022795号 | 川公网安备 51140202000112号