信号与信号量的区别[转]

2024-09-07 15:08
文章标签 信号量 区别 信号

本文主要是介绍信号与信号量的区别[转],希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

信号量(Semaphore),有时被称为信号灯,是在多环境下使用的一种设施,是可以用来保证两个或多个关键代码段不被并发调用。在进入一个关键代码段之前,线程必须获取一个信号量;一旦该关键代码段完成了,那么该线程必须释放信号量。其它想进入该关键代码段的线程必须等待直到第一个线程释放信号量。为了完成这个过程,需要创建一个信号量VI,然后将Acquire Semaphore VI以及Release Semaphore VI分别放置在每个关键代码段的首末端。确认这些信号量VI引用的是初始创建的信号量。

1.信号:(signal)是一种处理异步事件的方式。信号时比较复杂的通信方式,用于通知接受进程有某种事件发生,除了用于进程外,还可以发送信号给进程本身。linux除了支持unix早期的信号语义函数,还支持语义符合posix.1标准的信号函数sigaction。  

2.信号量:(Semaphore)进程间通信处理同步互斥的机制。是在多线程环境下使用的一种设施, 它负责协调各个线程, 以保证它们能够正确、合理的使用公共资源。 

  竞态条件(race condition)是一个在设备或者系统试图同时执行两个操作的时候出现的不希望的状况,但是由于设备和系统的自然特性,为了正确地执行,操作必须按照合适顺序进行。  在计算机内存或者存储里,如果同时发出读写大量数据的指令的时候竞态条件可能发生,机器试图覆盖相同的或者就的数据,而此时旧的数据仍然在被读取。结果可能是下面一个或者多个情况:计算机死机,出现非法操作提示并结束程序,错误的读取旧的数据,或者错误的写入新数据。在串行的内存和存储访问能防止这种情况,当读写命令同时发生的时候,默认是先执行读操作的。  如果两个或两个以上的线程同时访问相同的对象,或者访问不同步的共享状态.就会出现竞态条件。

 

转自:http://blog.csdn.net/yuzhoudiwang/article/details/4368279

现在最常用的进程间通信的方式有:信号,信号量,消息队列,共享内存。 
      所谓进程通信,就是不同进程之间进行一些"接触",这种接触有简单,也有复杂。机制不同,复杂度也不一样。通信是一个广义上的意义,不仅仅指传递一些massege。他们的使用方法是基本相同的,所以只要掌握了一种的使用方法,然后记住其他的使用方法就可以了。 

      1. 信号 
      在我学习的内容中,主要接触了信号来实现同步的机制,据说信号也可以用来做其它的事
      情,但是我还不知道做什么。 
      信号和信号量是不同的,他们虽然都可用来实现同步和互斥,但前者是使用信号处理器来
      进行的,后者是使用P,V*作来实现的。 
      使用信号要先知道有哪些信号,在Linux下有31个需要记住的通用信号,据说也是system
      V中最常用的那些。这里略。 
      1. 1信号相关函数: 
      #include 
      int sigaction(int signo, const struct sigaction *act, struct sigaction
      *oact); 
      该函数用来为进程安装信号处理器,struct sigaction数据是用来保存信号处理器的相
      关信息。 

      #include 
      int sigemptyset(sigset_t *set); 
      将信号集合清空。 
      int sigfillset(sigset_t *set); 
      将信号集合设置成包含所有的信号。在对信号进行*作以前一定要对信号集进行初始化。

      int sigaddset(sigset_t *set, int signo); 
      向信号集中加入signo对应的新信号。 
      int sigdelset(sigset_t *set, int signo); 
      从信号集中删除signo对应的一个信号。 
      int sigismember(const sigset_t *set, int signo); 
      判断某个信号是否在信号集中。返回1则在,0则不在。 

      #include 
      int sigprocmask(int how,const sigset_t *set, sigset_t *oset); 
      用来设置进程的信号屏蔽码。信号屏蔽码可以用来在某段时间内阻塞一些信号集中的信
      号,如果信号不在信号集中,就不必讨论它,因为肯定不响应,是否能生成也不肯定,我
      没有做过试验。 

      1.2我所理解的使用信号机制的方法: 
      使用信号,主要做的事情就是信号处理器的工作,这里面是你想做的事情。就像中断处理
      函数一样。 
      在使用信号以前,首先要初始化信号集,只有在信号集里面的信号才会被考虑。 
      有两种方法可以初始化信号集,一种是设置空信号集,一种是将所有的信号都加到信号集
      中。如果你自己想要的信号集不是这两种,可以在初始化了以后通过添加和删除信号进行
      定制。 
      如果在进程执行的一段时间内不想对某些信号进行响应,则可以使用sigprocmask对当前
      的信号集中的一些信号进行阻塞,稍后再执行。 
      当你将信号集设置完毕后,在让他工作之前需要安装信号处理器。安装信号处理器可以实
      现这几个功能: 
      指定信号处理函数的入口;指定信号屏蔽集合;指定信号处理器的一些标志。所谓信号处
      理器,就是指定了一些处理方法,关键在于安装信号处理器,这是使正确的信号进行正确
      的处理关键。在安装的时候,一定要对特定的信号赋予正确的信号处理函数。 
      我不知道不同进程之间的信号处理器能否混用,但是像一个特定的进程中有多少个信号处
      理器这样的问题是不能提的。因为信号处理器是一个概念,他针对的是信号,就是说如果
      你指定了一个数据结构,用它来存储针对某个信号的处理信息,那么安装信号处理器就是
      赋予这个数据结构一些相关信息,使用信号处理器就是用这个数据结构存储的信息来组织
      一种机制当发生这个信号的时候会做一些你实现设置好的处理。但是如果区分不同进程中
      对同一个信号的不同处理器?我想处理器可能只对核它所属的进程有关的信号进行响应,
      但是如果是这样的话,那这是怎么实现的呢? 
      不过有一点是可以知道的,那就是每一个信号都有一个信号处理器(确定的),可以动过
      安装信号处理器来指定她的行为。信号处理器由他自己的信息存储区域(我不知道在什么
      地方),但是可以通过向sigaction类型的数据结构向信号处理器的信息存储区域中传递
      信息。这个数据结构由一个就可以了,因为它只是临时传递数据的载体。 
      但是sigpromask和信号处理器里面的sigmask是不一样的,前者是在进程当前流程设置信
      号屏蔽,后者是指定在信号处理器作用时需要屏蔽掉的信号。例如,在设置某个特定信号
      的信号处理器时,我们当然不能让它的信号处理器工作了,因为还没有设置完吗,这是我
      们可以使用sigprocmask来让当前的流程开始阻塞该信号,当设置完信号处理器以后,再
      用sigprocmask恢复被阻塞的信号。而以后再接收到该信号时,信号处理器就可以工作了。 
      我的想法是,同一个信号在不同的进程里可以有不同的信号处理器(一般应该有一个缺省
      处理),当系统中发生一个信号时,所有能接受到的进程都可以接收到这个信号,并用他
      们自己的信号处理器对这个信号做出各自的响应。 

      1.3如何用信号来进行进程间的同步 
      同步的实现主要是通过在接受信号之前挂起进程,等待相关信号。所以涉及到异步信号安
      全函数的概念。 

      不过信号如何来实现进程间的互斥,我理解不是很多,我想信号的主要用处还是在软中断
      处理和进程同步。 


      2.信号量 
      信号量和信号是不同的东西,仔细想想就可以理解:信号是实现约定的固定的值,而信号
      量是一个变量记录着某些特定信息。 
      信号量这种东西我们在*作系统课程中就已经接触过了,这里只是再草草说几句。信号量
      分为有名和无名两种。进程间通信用有名信号量,同一进程内部通信一般用无名信号量。
      这个我不再多说。 
      2.1信号量相关函数 
      #include 
      #include 
      #include 
      int semget(key_t key, int nsems, int semflg); 
      创建一个新的信号量组或获取一个已经存在的信号量组。 

      #include 
      #include 
      #include 
      int semop(int semid, struct sembuf *sop, int nsops); 
      semop函数可以一次对一个或多个信号量进行*作。 
      Int semctl(int sem_id, int semnum, int cmd,/*union semun arg*/…); 
      该函数可以用来获取一些信号量的使用信息或者是来对信号量进行控制。 

      2.2我对信号量机制的理解 
      对信号量的*作只有两个:P, V。 
      为了在逻辑上便于组织信号量,信号量机制中有一个概念是信号量组。我们可以把一个信
      号量组中创建相关的信号量,这样逻辑上清晰也便于管理。在使用之前你同样需要对他们
      进行初始化:生成或打开信号量组,向其中生成或删除你指定的信号量。 
      对信号量的*作只用两种,他都是通过semop函数中的sops参数来指定的,如果这个参数
      是一个数组的话,那么就是对多个信号量进行*作。Sops参数中的sem_op字段指明了对信
      号量进行的是P*作还是V*作。你只要指定就行了,具体的*作不需要你去实现,函数中
      都已经提供了。使用信号量,你得清楚信号量组id和信号量在信号量组中的位置(其实也
      就是另一个id)。一个信号量必须属于一个信号量组,否则不能被系统所使用。切记! 
      信号量和信号量组是不会被系统所自动清理的,所以当你的进程退出前,千万别忘了清理
      你生成的那些信号量们。 
      信号量既可以实现互斥,也可以实现同步,这里就不说了,*作系统课程中是有介绍的。



      3.消息队列 
      消息队列是比较高级的一种进程间通信方法,因为它真的可以在进程间传送massege,你
      传送一个"I seek you"都可以。 
      一个消息队列可以被多个进程所共享(IPC就是在这个基础上进行的);如果一个进程的
      消息太多一个消息队列放不下,也可以用多于一个的消息队列(不过可能管理会比较复
      杂)。共享消息队列的进程所发送的消息中除了massege本身外还有一个标志,这个标志
      可以指明该消息将由哪个进程或者是哪类进程接受。每一个共享消息队列的进程针对这个
      队列也有自己的标志,可以用来声明自己的身份。 
      对于系统中的每一个消息队列,都有一个数据结构来代表它,这个数据结构是msqid_ds,
      这里略去不讲,在中可以看到它的原型。 

      3.1消息队列相关函数 
      使用消息队列之前,你要么获得这个消息队列,要么自己建立一个,否则是不能使用消息
      队列的(我觉得这都像是多余的话,请见谅)。当这个消息队列不再使用时,也一定要有
      一个进程来删除消息队列,系统是不会自动的清理消息队列和msgid_ds的。 

      Int msgget(key_t key, int msgflg); 
      获取一个存在的消息队列的ID,或者是根据跟定的权限创建一个消息队列。但是怎么样去
      删除这个消息队列,我还不十分清楚。 
      Int msgctl(int msqid, int cmd, struct msqid_ds *buf); 
      用来从msqid_ds中获取很多消息队列本身的信息。 
      Int msgsnd(int msqid, void *msgp, size_t msgsz, int msgflg); 
      用于向队列发送消息。 
      Int msgrcv(int msqid, void *msgp, size_t msgsz, long int msgtyp, int
      msgflg); 
      从队列中接收消息。 
      我这个文档里面对消息队列中的一些临界情况所述不多,因为这是我的小结,而非介绍。
      在GNU C库技术中可以看到它的详细介绍。

这篇关于信号与信号量的区别[转]的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1145395

相关文章

2.1/5.1和7.1声道系统有什么区别? 音频声道的专业知识科普

《2.1/5.1和7.1声道系统有什么区别?音频声道的专业知识科普》当设置环绕声系统时,会遇到2.1、5.1、7.1、7.1.2、9.1等数字,当一遍又一遍地看到它们时,可能想知道它们是什... 想要把智能电视自带的音响升级成专业级的家庭影院系统吗?那么你将面临一个重要的选择——使用 2.1、5.1 还是

Python中@classmethod和@staticmethod的区别

《Python中@classmethod和@staticmethod的区别》本文主要介绍了Python中@classmethod和@staticmethod的区别,文中通过示例代码介绍的非常详细,对大... 目录1.@classmethod2.@staticmethod3.例子1.@classmethod

Golan中 new() 、 make() 和简短声明符的区别和使用

《Golan中new()、make()和简短声明符的区别和使用》Go语言中的new()、make()和简短声明符的区别和使用,new()用于分配内存并返回指针,make()用于初始化切片、映射... 详细介绍golang的new() 、 make() 和简短声明符的区别和使用。文章目录 `new()`

Python中json文件和jsonl文件的区别小结

《Python中json文件和jsonl文件的区别小结》本文主要介绍了JSON和JSONL两种文件格式的区别,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下... 众所周知,jsON 文件是使用php JSON(JavaScripythonpt Object No

无线路由器哪个品牌好用信号强? 口碑最好的三个路由器大比拼

《无线路由器哪个品牌好用信号强?口碑最好的三个路由器大比拼》不同品牌在信号覆盖、稳定性和易用性等方面各有特色,如何在众多选择中找到最适合自己的那款无线路由器呢?今天推荐三款路由器让你的网速起飞... 今天我们来聊聊那些让网速飞起来的路由器。在这个信息爆炸的时代,一个好路由器简直就是家庭网编程络的心脏。无论你

结构体和联合体的区别及说明

《结构体和联合体的区别及说明》文章主要介绍了C语言中的结构体和联合体,结构体是一种自定义的复合数据类型,可以包含多个成员,每个成员可以是不同的数据类型,联合体是一种特殊的数据结构,可以在内存中共享同一... 目录结构体和联合体的区别1. 结构体(Struct)2. 联合体(Union)3. 联合体与结构体的

电脑显示hdmi无信号怎么办? 电脑显示器无信号的终极解决指南

《电脑显示hdmi无信号怎么办?电脑显示器无信号的终极解决指南》HDMI无信号的问题却让人头疼不已,遇到这种情况该怎么办?针对这种情况,我们可以采取一系列步骤来逐一排查并解决问题,以下是详细的方法... 无论你是试图为笔记本电脑设置多个显示器还是使用外部显示器,都可能会弹出“无HDMI信号”错误。此消息可能

什么是 Ubuntu LTS?Ubuntu LTS和普通版本区别对比

《什么是UbuntuLTS?UbuntuLTS和普通版本区别对比》UbuntuLTS是Ubuntu操作系统的一个特殊版本,旨在提供更长时间的支持和稳定性,与常规的Ubuntu版本相比,LTS版... 如果你正打算安装 Ubuntu 系统,可能会被「LTS 版本」和「普通版本」给搞得一头雾水吧?尤其是对于刚入

python中json.dumps和json.dump区别

《python中json.dumps和json.dump区别》json.dumps将Python对象序列化为JSON字符串,json.dump直接将Python对象序列化写入文件,本文就来介绍一下两个... 目录1、json.dumps和json.dump的区别2、使用 json.dumps() 然后写入文

native和static native区别

本文基于Hello JNI  如有疑惑,请看之前几篇文章。 native 与 static native java中 public native String helloJni();public native static String helloJniStatic();1212 JNI中 JNIEXPORT jstring JNICALL Java_com_test_g