Linux编程练习 --多线程5--信号量(semaphore)

这一篇练习信号量的应用

信号量本质上是一个非负的整数计数器,也是UNIX中古老的实现进程互斥和同步的手段,Linux下信号量概念是在线程中,信号则在进程控制中,不过原理差不多,最基本最经典的操作莫过于P、V操作了,能实现进程、线程的互斥和同步操作,非常值得深入理解。

1.P、V操作原语

 P操作:

 proceduce P(var s:semaphore)

{

  begin

      s:=s-1;

      if(s<0)then W(s);

  end

}

 

V操作:

 proceduce V(var s:semaphore)

{

  begin

      s:=s+1;

      if(s<=0)then R(s);

  end

 

2.基本操作

数据类型:信号量的数据类型为结构sem_t,它本质上是一个长整型的数。

函数:

(1)sem_init

功能:         用于创建一个信号量,并初始化信号量的值。

头文件:       <semaphore.h>

函数原型:     int sem_init (sem_t* sem, int pshared, unsigned int value);

函数传入值:   sem:信号量。

                   pshared:决定信号量能否在几个进程间共享。由于目前LINUX还没有实现进

                               程间共享信息量,所以这个值只能取0。

                   value:初始计算器

函数返回值:   0:成功。

                   -1:失败。

(2)其他函数。

//等待信号量

int sem_wait       (sem_t* sem);

int sem_trywait   (sem_t* sem);

//发送信号量

int sem_post       (sem_t* sem);

//得到信号量值

int sem_getvalue (sem_t* sem);

//删除信号量

int sem_destroy   (sem_t* sem);

功能:sem_wait和sem_trywait相当于P操作,它们都能将信号量的值减一,两者的区别在

        于若信号量的值小于零时,sem_wait将会阻塞进程,而sem_trywait则会立即返回。

        sem_post相当于V操作,它将信号量的值加一,同时发出唤醒的信号给等待的进程

       (或线程)。

        sem_getvalue 得到信号量的值。

        sem_destroy 摧毁信号量。

函数传入值: sem:信号量。

函数返回值: 同上。

好了,了解完基本操作,继续做一个练习:

这里用信号量实现互斥资源访问的功能:

  1. /*sem.c*/  
  2. #include <stdlib.h>    
  3. #include <stdio.h>    
  4. #include <pthread.h>    
  5. #include <semaphore.h>  
  6. #include <errno.h>     
  7. /*全局变量*/  
  8. int gnum = 0;  
  9. /*信号量*/  
  10. sem_t sem;  
  11. /*声明线程运行服务程序*/  
  12. static void pthread_func_1 (void);     
  13. static void pthread_func_2 (void);     
  14.     
  15. int main (void)     
  16. {     
  17.  /*线程的标识符*/  
  18.   pthread_t pt_1 = 0;     
  19.   pthread_t pt_2 = 0;     
  20.   int ret = 0;     
  21.   /*信号量初始化*/  
  22.   sem_init(&sem,0,1);  
  23.   /*分别创建线程1、2*/  
  24.   ret = pthread_create (&pt_1,          //线程标识符指针  
  25.                          NULL,          //默认属性  
  26.                         (void *)pthread_func_1,//运行函数  
  27.                         NULL);          //无参数  
  28.   if (ret != 0)     
  29.   {     
  30.      perror ("pthread_1_create");     
  31.   }     
  32.     
  33.   ret = pthread_create (&pt_2,          //线程标识符指针  
  34.                         NULL,           //默认属性    
  35.                         (void *)pthread_func_2, //运行函数  
  36.                         NULL);          //无参数  
  37.   if (ret != 0)     
  38.   {     
  39.      perror ("pthread_2_create");     
  40.   }     
  41.   /*等待线程1、2的结束*/  
  42.   pthread_join (pt_1, NULL);     
  43.   pthread_join (pt_2, NULL);     
  44.     
  45.   printf ("main programme exit!/n");    
  46.   return 0;     
  47. }     
  48. /*线程1的服务程序*/  
  49. static void pthread_func_1 (void)     
  50. {     
  51.   int i = 0;     
  52.        
  53.   for (;;)     
  54.   {     
  55.     printf ("This is pthread1!/n");      
  56.     sem_wait(&sem);     /*等待信号量*/     
  57.     sleep (1);   
  58.     /*临界资源*/  
  59.     gnum++;  
  60.     printf ("Thread1 add one to num:%d/n",gnum);  
  61.       
  62.     sem_post (&sem);        /*释放信号量*/  
  63.   
  64.        
  65.   }     
  66. }     
  67. /*线程2的服务程序*/   
  68. static void pthread_func_2 (void)     
  69. {     
  70.   int i = 0;     
  71.     
  72.   for (;;)     
  73.   {     
  74.     printf ("This is pthread2!/n");   
  75.     sem_wait(&sem);     /*等待信号量*/  
  76.     sleep (1);  
  77.     /*临界资源*/  
  78.     gnum++;  
  79.     printf ("Thread2 add one to num:%d/n",gnum);  
  80.       
  81.     sem_post (&sem);        /*释放信号量*/  
  82.     
  83.   }     
  84.     
  85.   pthread_exit (0);     
  86. }    

编译,运行,可以看出和上次互斥锁结果一样的

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/436282.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【转】CAN协议深度解析-简单易懂协议详解

转自&#xff1a;https://zhuanlan.zhihu.com/p/343607068 CAN接口兼容规范2.0A和2.0B(主动)&#xff0c;位速率高达1兆位/秒。它可以接收和发送11位标识符的标准帧&#xff0c;也可以接收和发送29位标识符的扩展帧。 扩展帧的仲裁域有29位&#xff0c;可以出现2^29中报文&…

深度神经网络下的风格迁移模型(C#)

版权声明&#xff1a;本文为博主原创文章&#xff0c;转载请在显著位置标明本文出处以及作者网名&#xff0c;未经作者允许不得用于商业目的。 这个是C#版本的&#xff0c;这里就只放出代码。VB.Net版本请参看 深度神经网络下的风格迁移模型-CSDN博客 斯坦福大学李飞飞团队的…

使用 Visual Studio 2005 Team System 进行单元测试并生成用于 Unit Test Framework 的源代码...

PS&#xff1a;微软专家写的一往篇VSTS2005测试功能说明&#xff0c;非常详细。适合初学者查看。适用于&#xff1a;Microsoft Visual Studio 2005 Team System Beta 2Team Architect & Team Test EditionsMicrosoft Visual C# 2005摘要&#xff1a; Scott 详细介绍自动化单…

java银行利率_Java-银行计算利息

Bank类public class Bank {int savedMoney;int year;double interest;double interestRate0.29;public double computerInterest(){interestyear*interestRate*savedMoney;return interest;}public void setInterestRate(double rate){interestRaterate;}}ConstructionBank类pu…

【转】C++ win32窗口创建详解

转自&#xff1a;https://my.oschina.net/u/4328928/blog/3315324 本篇所讲解的内容仅限于 Windows 操作系统且限于 win32程序设计 现在我们在Windows系统上用的软件, 早已不是控制台界面, 而是窗体应用程序 窗体与控制台的区别就是: 有了窗口的概念 由于C的语法复杂, 使得…

java 子类继承父类_关于Java 的继承问题,子类会继承父类的哪些东西?-----转载...

和C类似&#xff0c;可以继承基类的公共属性和方法。在Java继承里&#xff0c;父类的属性还有方法在声明时&#xff0c;如果是public关键字即公共属性&#xff0c;则在子类继承时&#xff0c;这些属性和方法都会被子类继承。受保护的也可以继承但是私有的类属性成员和方法则无法…

天气很冷晚饭很香

今天和zhoujia 去吃她家门口的东北菜&#xff0c;席间谈笑风生&#xff0c;突然她说“亲爱的&#xff0c;眼光要看的长远一些”&#xff0c;附带标准的领导手势&#xff0c;让我笑翻。zhoujia是个乐观的人&#xff0c;从来没有见她发愁过&#xff0c;她的房子装修的也差不多了&…

【转】win32窗口的大小,居中,拖动

参考 https://www.cnblogs.com/findumars/p/3948315.html 不让调整大小 窗口风格设置&#xff0c;后面两个是最大最小化按钮 dwStyle^WS_THICKFRAME^WS_MAXIMIZEBOX^WS_MINIMIZEBOX 不让拖动 消息循环里添加 case WM_NCLBUTTONDOWN: { switch (wParam) …

java +号变空格_base64码通过http传输 +号变 空格 问题解决

通过七牛云base64上传图片&#xff0c;通过官方示例上传成功后&#xff0c;根据示例改了一个controller。通过前端往后端传base64码形式进行测试。死活不通过&#xff0c;七牛报400。仔细排查后发现&#xff0c;示例转换的base64码与前端传来的base64码稍有区别&#xff0c;前端…

Linux编程练习 --进程间通信1--无名管道

进程间通信系列--管道 管道可以说是最古老的IPC形式&#xff0c;所谓管道&#xff0c;是指进程间建立的一条通信的通道&#xff0c;从本质上看&#xff0c;管道是UNIX文件概念的推广管道通信的介质是文件&#xff0c;先看一下管道的特点&#xff1a; 1.管道的特点&#xff1a; …

2.3 Factory Method(工厂方法)

【返回目录】 我们现在把场景从险象丛生的特工战切换到更为壮观的二战中来&#xff0c;很多人都看过《拯救大兵瑞恩》&#xff0c;这部电影最著名的两场战斗非首和尾莫属&#xff0c;我们就用最后一场战役来举例说明什么是工厂方法吧。 相比诺曼底登陆&#xff0c;最后一场战役…

【转】CreateWindow函数详解

CreateWindow函数详解 在注册完窗口类后就需要进行窗口的创建&#xff0c;用到的函数理所当然就是CreateWindow()&#xff0c; 而这个函数是基于窗口类的&#xff0c;所以还需要指定几个参数来制定特定的窗口。而且像一些不带边框的窗口是怎么创建的也是具有相当的技巧的&#…

java中处理打折率_【JAVA300例】13、输入价格判断折扣,switch用法+int留整数方便判断...

import java.util.Scanner;public class Test013{public static void main(String[] args){Scanner in new Scanner(System.in);System.out.println("请输入金额(整数):");float money in.nextFloat();String zhekou "";if (money>1200){int grade …

[转] UML中关联、依赖、聚集等关系的异同

1. 关联&#xff1a;连接模型元素及链接实例&#xff0c;用一条实线来表示&#xff1b;2. 依赖&#xff1a;表示一个元素以某种方式依赖于另一个元素&#xff0c;用一条虚线加箭头来表示&#xff1b;3. 聚集&#xff1a;表示整体与部分的关系&#xff0c;用一条实线加空心菱…

【转】C++ _T()函数和_L()函数介绍

_T("")是一个宏,他的作用是让你的程序支持Unicode编码&#xff0c;因为Windows使用两种字符集ANSI和UNICODE&#xff0c; 前者就是通常使用的单字节方式&#xff0c;但这种方式处理象中文这样的双字节字符不方便&#xff0c;容易出现半个汉字的情况。 而后者是双字节…

Linux编程练习 --进程间通信2--两个管道实现双向通信

利用两个管道进行进程间双向通信在第一篇练习已经大致作出说明&#xff0c;下面将进行一个更为综合的练习 首先看题目&#xff1a; 设有二元函数f(x,y) f(x) f(y) 其中&#xff1a; f(x) f(x-1) * x (x >1) f(x)1 (x1) f(y) f(y-1) f(y-2) (y> 2) f(y)1 (y1,2) 请编…

Windows集群网络配置最佳做法

要开始做集群了&#xff0c;找点资料来看看&#xff1a;阅读提示&#xff1a;本文为 Microsoft Windows 2000 或 Windows Server 2003 服务器群集的网络基础结构提供了服务器群集要求和最佳做法。若要群集可以正常运行&#xff0c;必须满足这些要求。最佳做法是从部署反馈和现场…

JOJ的2042面试题目的数学推导过程

JOJ的2042题目是一个程序理解题目&#xff0c;这个题目非常有意思&#xff0c;给出了下面一段C源代码&#xff0c;要求计算出最后的输出结果&#xff0c;源代码如下&#xff1a; #include<cstdio> int main(void) { int x 987654321, c 0, d 1, e 6; while…

linux mysql密码转义_linux忘记mysql密码处理方法

linux忘记mysql密码处理方法:# /etc/init.d/mysql stop# mysqld_safe --usermysql --skip-grant-tables --skip-networking &# mysql -u root mysqlmysql> update user set passwordpassword(newpassword) where userroot;mysql> flush privileges;mysql> quit# /…

不同职业的面试着装技巧。

美女姜培琳&#xff1a;传授独家心经 不同职业的面试着装技巧。 一般来说&#xff0c;职场中精英女性的装扮&#xff0c;首要应讲求端庄、稳重。人们对服饰过于花哨怪异者的工作能力、工作作风、敬业精神、生活态度等&#xff0c;都会持怀疑的态度。 其实&#x…