线程同步之信号量同步




linux中两种基本的同步方法是信号量和互斥量。这两种方法很相似,而且它们可以相互通过对方来实现。


现在有个图书馆,其能容纳100人,现在有两个线程A、B,A线程执行:往图书管理进入一个人,B线程:从图书馆出来一个人。那么为了使得线程A在图书馆满人的时候进入等待,而不是继续往图书馆里进人,使得B线程在图书馆没人的时候等待人进入,我们可以引入信号量:IN  OUT  分别初始化为100和0


   那么A则可以被表示为A:P(IN) //剩余容量减少一 如果容量为0 则等待                     B:P(OUT) //人数减少1  如果人数为0则阻塞等待

                                        .........   //登记进入图书馆的人信息                  .............  //记录 离开图书馆的人信息   (随便一系列操作。)

                                        V(OUT)//增加信号量OUT 表示人数+1                           V(IN)   //增加图书馆剩余容量+1

         通过这样我们就实现了线程的同步。


信号量      

 

  作用域 上锁时
信号量 进程间或线程间(linux仅线程间) 只要信号量的value大于0,其他线程就可以sem_wait成功,成功后信号量的value减一。若value值不大于0,则sem_wait阻塞,直到sem_post释放后value值加一
互斥锁 线程间 只要被锁住,其他任何线程都不可以访问被保护的资源 
成功后否则就阻塞


信号量概述


下面介绍用信号量进行同步。


信号量概念由荷兰科学家Dijkstra首先提出。信号量是一个特殊类型的变量,它可以被增加或者减少。但对其的关键访问被保证是原子操作,即使在一个多线程程序中也是如此。


信号量有两种类型:

(1)二进制信号量。它只有0和1两种取值

(2)计数信号量。它可以有更大的取值范围。

如果要用信号量来保护一段代码,使其每次只能被一个执行线程运行,就要用到二进制信号量、。

如果要允许有限数目的线程执行一段指定的代码,就需要用到计数信号量。


由于计数信号量并不常用,而且它实际上仅仅是二进制信号量的一种扩展,这里之介绍二进制信号量。


信号量的相关函数

信号量函数的名字都以sem_开头。线程中使用的基本函数有4个。

注意,需要包含头文件:

  1. #include<semaphore.h>  

创建信号量

  1. int sem_init(sem_t *sem, int pshared, unsigned int value);  
函数解释:

sem_init() 初始化一个定位在 sem 的匿名信号量value 参数指定信号量的初始值。pshared 参数指明信号量是由进程内线程共享,还是由进程之间共享。如果 pshared 的值为 0,那么信号量将被进程内的线程共享,并且应该放置在所有线程都可见的地址上(如全局变量,或者堆上动态分配的变量)。
  如果 pshared 是非零值,那么信号量将在进程之间共享,并且应该定位共享内存区域(见 shm_open(3)、mmap(2) 和 shmget(2))。(因为通过 fork(2) 创建的孩子继承其父亲的内存映射,因此它也可以见到这个信号量。)所有可以访问共享内存区域的进程都可以使用sem_post(3)、sem_wait(3) 等等操作信号量。初始化一个已经初始的信号量其结果未定义。

返回值
  sem_init() 成功时返回 0;错误时,返回 -1,并把 errno 设置为合适的值。
错误
  EINVAL
  value 超过 SEM_VALUE_MAX。
  ENOSYS
  pshared 非零,但系统还没有支持进程共享的信号量。


下面是控制信号量的两个函数:

信号量减一操作

  1. int sem_wait(sem_t * sem);  
函数说明
  sem_wait函数也是一个原子操作,它的作用是从信号量的值减去一个“1”,但它永远会先等待该信号量为一个非零值才开始做减法。也就是说,如果你对一个值为2的信号量调用sem_wait(),线程将会继续执行,这信号量的值将减到1。如果对一个值为0的信号量调用sem_wait(),这个函数就 会地等待直到有其它线程增加了这个值使它不再是0为止。如果有两个线程都在sem_wait()中等待同一个信号量变成非零值,那么当它被第三个线程增加 一个“1”时,等待线程中只有一个能够对信号量做减法并继续执行,另一个还将处于等待状态。
返回值
  所有这些函数在成功时都返回 0;错误保持信号量值没有更改,-1 被返回,并设置 errno 来指明错误。
错误
  EINTR
  这个调用被信号处理器中断,
  EINVAL

  sem 不是一个有效的信号量。


信号量加一操作

  1. int sem_post(sem_t * sem);   
说明
  sem_post函数的作用是给信号量的值加上一个“1”,它是一个“原子操作”---即同时对同一个信号量做加“1”操作的两个线程是不会冲突的;而同 时对同一个文件进行读、加和写操作的两个程序就有可能会引起冲突。信号量的值永远会正确地加一个“2”--因为有两个线程试图改变它。
返回值
  sem_post() 成功时返回 0;错误时,信号量的值没有更改,-1 被返回,并设置 errno 来指明错误。
错误
  EINVAL
  sem 不是一个有效的信号量。
  EOVERFLOW
  信号量允许的最大值将要被超过。


清理信号量

  1. int sem_destroy (sem_t *sem);  
 这个函数也使用一个信号量指针做参数,归还自己战胜的一切资源。在清理信号量的时候如果还有线程在等待它,用户就会收到一个错误。
        与其它的函数一样,这些函数在成功时都返回“0”。


信号量的使用

下面主线程中创建了一个新线程,用来统计输入的字符串中字符的个数。信号量用来控制两个线程对存储字符串数组的访问

(一个线程用来控制输入,另一个线程用来控制输出)

代码:

  1. #include <stdio.h>  
  2. #include <unistd.h>  
  3. #include <stdlib.h>  
  4. #include <string.h>  
  5. #include <pthread.h>  
  6. #include <semaphore.h>  
  7.   
  8. //线程函数   
  9. void *thread_function(void *arg);  
  10.   
  11. sem_t bin_sem;//信号量对象  
  12.   
  13. #define WORK_SIZE 1024  
  14.   
  15. char work_area[WORK_SIZE];//工作区  
  16.   
  17. int main()  
  18. {  
  19.     int res;  
  20.     pthread_t a_thread;  
  21.     void *thread_result;  
  22.   
  23.     res = sem_init(&bin_sem,0,0);//初始化信号量对象  
  24.   
  25.     if(res)//初始化信号量失败  
  26.     {  
  27.     perror("Semaphore initialization failed\n");  
  28.     exit(EXIT_FAILURE);  
  29.     }  
  30.   
  31.     //创建新线程  
  32.     res = pthread_create(&a_thread,NULL,thread_function,NULL);  
  33.     if(res)  
  34.     {  
  35.     perror("Thread creation failed\n");  
  36.     exit(EXIT_FAILURE);  
  37.     }  
  38.   
  39.     printf("Input some text.Enter 'end' to finish\n");  
  40.   
  41.     while(strncmp("end",work_area,3) != 0)  
  42.     {//输入没有结束  
  43.     fgets(work_area,WORK_SIZE,stdin);  
  44.     sem_post(&bin_sem);//给信号量值加一  
  45.     }  
  46.   
  47.     printf("waiting for thread to finish\n");  
  48.   
  49.     //等待子线程结束,收集子线程信息  
  50.     res = pthread_join(a_thread,&thread_result);  
  51.     if(res)  
  52.     {  
  53.     perror("Thread join failed\n");  
  54.     exit(EXIT_FAILURE);  
  55.     }  
  56.   
  57.     printf("Thread joined\n");  
  58.   
  59.     //销毁信号量对象  
  60.     sem_destroy(&bin_sem);  
  61.   
  62.     exit(EXIT_SUCCESS);  
  63. }  
  64.   
  65. void *thread_function(void *arg)  
  66. {  
  67.     sem_wait(&bin_sem);//将信号量值减一。  
  68.   
  69.     while(strncmp("end",work_area,3))  
  70.     {  
  71.         printf("You input %d characters\n",strlen(work_area) - 1);  
  72.         sem_wait(&bin_sem);  
  73.     }  
  74.   
  75.     pthread_exit(NULL);//线程终止执行  
  76. }  

运行结果:

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值