百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 技术分析 > 正文

Linux信号量详解 linux 信号

liebian365 2024-11-03 15:46 22 浏览 0 评论

信号量

1、信号量和P、V原语

信号量和P、V原语由迪杰斯特拉提出

互斥:P、V在同一个进程中

同步:P、V在不同进程中

信号量值含义:

(1)S>0: S表示可用资源的个数。

(2)S=0:表示无可用资源,无等待进程。

(3)S<0: |S|表示等待队列中进程个数。

P原语:

P(s)
{
s.value = s.value--;
if(s.value < 0)
{
//该进程状态置为等待状态
//将该进程的PCB插入相应的等待队列s.queue末尾
}
}

V原语:

V(s)
{
s.value = s.value++;
if(s.value <= 0)
{
//唤醒相应等待队列s.queue中等待的一个进程
//改变其状态为就绪态,并将其插入就绪队列
}
}

2、信号量集函数

//(1)semget函数
// 功能:用来创建和访问一个信号量集
//原型:
int semget(key_t key,int nsems,int semflg);
//参数: key:信号集的名字
// nsems:信号集中信号量的个数
// semflg:由九个权限标志构成,它们的用法和创建文件时使用的mode模式标志是一样的
//返回值:成功返回0,失败返回-1
//(2)shmctl函数
// 功能:用于控制信号量集
//原型:
int semctl(int semid,int semnum,int cmd, ...);
//参数: semid:由semget返回的信号集标识码
// semnum:信号集中信号量的个数
// cmd:将要采取的动作(有三个可取值)
// 最后一个参数根据不同而不同
//返回值:成功返回0,失败返回-1
//(3)semop函数
//功能:用来创建和访问一个信号量集
//原型:
int semop(int semid,struct sembuf* sops,unsigned nsops);
//参数:
// semid:是该信号量的标识码
// sops:是个指向一个结构数值的指针
// nsops:信号量个数
//返回值:成功返回0,失败返回-1

3、实例代码

注:

sembuf结构体

struct sembuf{
short sem_num; //信号量的编号
short sem_op; //是信号量一次PV操作时加减的数值,一般只会有两个 -1(P操作) 或 +1(V操作)
short sem_flg; //有两个取值是IPC_NOWAIT或SEM_UNDO
}
//【comm.h】
#ifndef _COMM_H__
#define _COMM_H__
#include#include#include#include#define PATHNAME "."
#define PROJ_ID 0x6666
union semun
{
 int val;
 struct semid_ds* buf;
 unsigned short* array;
 struct seminfo* _buf;
};
int createSemSet(int nums);
int initSem(int semid,int nums,int initval);
int getSemSet(int nums);
int P(int semid,int who);
int V(int semid,int who);
int destorySemSet(int semid);
#endif
//【comm.c】
#include"comm.h"
static int commSemSet(int nums,int flags)
{
 key_t _key=ftok(PATHNAME,PROJ_ID);
 if(_key<0)
 {
 perror("ftok");
 return -1;
 }
 int semid=semget(_key,nums,flags);
 if(semid<0)
 {
 perror("semget");
 return -2;
 }
 return semid;
}
int createSemSet(int nums)
{
 return commSemSet(nums,IPC_CREAT|IPC_EXCL|0666);
}
int getSemSet(int nums)
{
 return commSemSet(nums,IPC_CREAT);
}
int initSem(int semid,int nums,int initval)
{
 union semun _un;
 _un.val=initval;
 if(semctl(semid,nums,SETVAL,_un)<0)
 {
 perror("semctl");
 return -1;
 }
 return 0;
}
static int commPV(int semid,int who,int op)
{
 struct sembuf _sf;
 _sf.sem_num=who;
 _sf.sem_op=op;
 _sf.sem_flg=0;
 if(semop(semid,&_sf,1)<0)
 {
 perror("semop");
 return -1;
 }
 return 0;
}
int P(int semid,int who)
{
 return commPV(semid,who,-1);
}
int V(int semid,int who)
{
 return commPV(semid,who,1);
}
int destorySemSet(int semid)
{
 if(semctl(semid,0,IPC_RMID)<0)
 {
 perror("semctl");
 return -1;
 }
 return 0;
}
//【sem_test.c】
#include"comm.h"
int main()
{
 int semid=createSemSet(1);
 printf("se=%d\n",semid);
 initSem(semid,0,1);
 pid_t id=fork();
 if(id==0)
 {
 //child
 int _semid=getSemSet(0);
 printf("_semid=%d\n",_semid);
 while(1)
 {
 P(_semid,0);
 printf("A");
 fflush(stdout);
 usleep(200000);
 printf("A ");
 fflush(stdout);
 usleep(200000);
 V(_semid,0);
 }
 }
 else
 {
 //father
 while(1)
 {
 P(semid,0);
 printf("B");
 fflush(stdout);
 usleep(200000);
 printf("B ");
 fflush(stdout);
 usleep(200000);
 V(semid,0);
 }
 wait(NULL);
 }
 destorySemSet(semid);
 return 0;
}

运行结果:

将comm.c封装成静态库:

[lize-h@localhost 0406_SignalNum]$ ls
a.out comm.c comm.h comm.o Makefile test_sem.c
[lize-h@localhost 0406_SignalNum]$ gcc -c comm.c -o comm.o 
[lize-h@localhost 0406_SignalNum]$ ls
a.out comm.c comm.h comm.o Makefile test_sem.c
生成静态库:
[lize-h@localhost 0406_SignalNum]$ ar -rc libmycomm.a comm.o
ar是gnu归档工具,rc表示(replace and create)
[lize-h@localhost 0406_SignalNum]$ ar -tv libmycomm.a
rw-rw-r-- 500/500 1676 May 4 21:16 2018 comm.o
t:列出静态库中的文件
v:verbose详细信息
在不调用静态库的情况下编译失败
[lize-h@localhost 0406_SignalNum]$ gcc test_sem.c
/tmp/ccvjqyzV.o: In function `main':
test_sem.c:(.text+0x11): undefined reference to `createSemSet'
test_sem.c:(.text+0x46): undefined reference to `initSem'
test_sem.c:(.text+0x66): undefined reference to `getSemSet'
test_sem.c:(.text+0x93): undefined reference to `P'
test_sem.c:(.text+0xf2): undefined reference to `V'
test_sem.c:(.text+0x108): undefined reference to `P'
test_sem.c:(.text+0x167): undefined reference to `V'
collect2: ld 返回 1
调用静态库
[lize-h@localhost 0406_SignalNum]$ gcc test_sem.c -L. -lmycomm -o comm
-L :指定库路径
-l :指定库名
测试目标文件生成后,删除静态库程序照样可以运行
[lize-h@localhost 0406_SignalNum]$ ls
comm comm.c comm.h comm.o libmycomm.a Makefile test_sem.c

将comm.c封装成动态库:

[lize-h@localhost 0406_SignalNum]$ ls
comm.c comm.h comm.o libmycomm.a Makefile test_sem.c
shared:表示生成共享库
fPIC:产生位置无关码
[lize-h@localhost 0406_SignalNum]$ gcc -fPIC -c comm.c
[lize-h@localhost 0406_SignalNum]$ gcc -shared -o libmycomm.so *.o
[lize-h@localhost 0406_SignalNum]$ ls
comm.c comm.h comm.o libmycomm.a libmycomm.so Makefile test_sem.c
调用动态库进行编译
[lize-h@localhost 0406_SignalNum]$ gcc test_sem.c -L. -lmycomm
[lize-h@localhost 0406_SignalNum]$ ls
a.out comm.c comm.h comm.o libmycomm.a libmycomm.so Makefile test_sem.c
[lize-h@localhost 0406_SignalNum]$ ./a.out
se=131074
B_semid=131074
B AA BB A^C
[lize-h@localhost 0406_SignalNum]$
//可以将 .so文件拷贝到系统共享库路径下,通常为 /usr/lib
将.so文件放到系统共享路径下调用时更简单
[lize-h@localhost 0406_SignalNum]$ gcc test_sem.c -lmycomm

相关推荐

“版本末期”了?下周平衡补丁!国服最强5套牌!上分首选

明天,酒馆战棋就将迎来大更新,也聊了很多天战棋相关的内容了,趁此机会,给兄弟们穿插一篇构筑模式的卡组推荐!老规矩,我们先来看10职业胜率。目前10职业胜率排名与一周前基本类似,没有太多的变化。平衡补丁...

VS2017 C++ 程序报错“error C2065:“M_PI”: 未声明的标识符&quot;

首先,程序中头文件的选择,要选择头文件,在文件中是没有对M_PI的定义的。选择:项目——>”XXX属性"——>配置属性——>C/C++——>预处理器——>预处理器定义,...

东营交警实名曝光一批酒驾人员名单 88人受处罚

齐鲁网·闪电新闻5月24日讯酒后驾驶是对自己和他人生命安全极不负责的行为,为守护大家的平安出行路,东营交警一直将酒驾作为重点打击对象。5月23日,东营交警公布最新一批饮酒、醉酒名单。对以下驾驶人醉酒...

Qt界面——搭配QCustomPlot(qt platform)

这是我第一个使用QCustomPlot控件的上位机,通过串口精确的5ms发送一次数据,再将读取的数据绘制到图表中。界面方面,尝试卡片式设计,外加QSS简单的配了个色。QCustomPlot官网:Qt...

大话西游2分享赢取种族坐骑手办!PK趣闻录由你书写

老友相聚,仗剑江湖!《大话西游2》2021全民PK季4月激燃打响,各PK玩法鏖战齐开,零门槛参与热情高涨。PK季期间,不仅各种玩法奖励丰厚,参与PK趣闻录活动,投稿自己在PK季遇到的趣事,还有机会带走...

测试谷歌VS Code AI 编程插件 Gemini Code Assist

用ClaudeSonnet3.7的天气测试编码,让谷歌VSCodeAI编程插件GeminiCodeAssist自动编程。生成的文件在浏览器中的效果如下:(附源代码)VSCode...

顾爷想知道第4.5期 国服便利性到底需优化啥?

前段时间DNF国服推出了名为“阿拉德B计划”的系列改版计划,截至目前我们已经看到了两项实装。不过关于便利性上,国服似乎还有很多路要走。自从顾爷回归DNF以来,几乎每天都在跟我抱怨关于DNF里面各种各样...

掌握Visual Studio项目配置【基础篇】

1.前言VisualStudio是Windows上最常用的C++集成开发环境之一,简称VS。VS功能十分强大,对应的,其配置系统较为复杂。不管是对于初学者还是有一定开发经验的开发者来说,捋清楚VS...

还嫌LED驱动设计套路深?那就来看看这篇文章吧

随着LED在各个领域的不同应用需求,LED驱动电路也在不断进步和发展。本文从LED的特性入手,推导出适合LED的电源驱动类型,再进一步介绍各类LED驱动设计。设计必读:LED四个关键特性特性一:非线...

Visual Studio Community 2022(VS2022)安装图文方法

直接上步骤:1,首先可以下载安装一个VisualStudio安装器,叫做VisualStudioinstaller。这个安装文件很小,很快就安装完成了。2,打开VisualStudioins...

Qt添加MSVC构建套件的方法(qt添加c++11)

前言有些时候,在Windows下因为某些需求需要使用MSVC编译器对程序进行编译,假设我们安装Qt的时候又只是安装了MingW构建套件,那么此时我们该如何给现有的Qt添加一个MSVC构建套件呢?本文以...

Qt为什么站稳c++GUI的top1(qt c)

为什么现在QT越来越成为c++界面编程的第一选择,从事QT编程多年,在这之前做C++界面都是基于MFC。当时为什么会从MFC转到QT?主要原因是MFC开发界面想做得好看一些十分困难,引用第三方基于MF...

qt开发IDE应该选择VS还是qt creator

如果一个公司选择了qt来开发自己的产品,在面临IDE的选择时会出现vs或者qtcreator,选择qt的IDE需要结合产品需求、部署平台、项目定位、程序猿本身和公司战略,因为大的软件产品需要明确IDE...

Qt 5.14.2超详细安装教程,不会来打我

Qt简介Qt(官方发音[kju:t],音同cute)是一个跨平台的C++开库,主要用来开发图形用户界面(GraphicalUserInterface,GUI)程序。Qt是纯C++开...

Cygwin配置与使用(四)——VI字体和颜色的配置

简介:VI的操作模式,基本上VI可以分为三种状态,分别是命令模式(commandmode)、插入模式(Insertmode)和底行模式(lastlinemode),各模式的功能区分如下:1)...

取消回复欢迎 发表评论: