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

Linux信号量详解 linux 信号

liebian365 2024-11-03 15:46 6 浏览 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

相关推荐

快递查询教程,批量查询物流,一键管理快递

作为商家,每天需要查询许许多多的快递单号,面对不同的快递公司,有没有简单一点的物流查询方法呢?小编的回答当然是有的,下面随小编一起来试试这个新技巧。需要哪些工具?安装一个快递批量查询高手快递单号怎么快...

一键自动查询所有快递的物流信息 支持圆通、韵达等多家快递

对于各位商家来说拥有一个好的快递软件,能够有效的提高自己的工作效率,在管理快递单号的时候都需要对单号进行表格整理,那怎么样能够快速的查询所有单号信息,并自动生成表格呢?1、其实方法很简单,我们不需要一...

快递查询单号查询,怎么查物流到哪了

输入单号怎么查快递到哪里去了呢?今天小编给大家分享一个新的技巧,它支持多家快递,一次能查询多个单号物流,还可对查询到的物流进行分析、筛选以及导出,下面一起来试试。需要哪些工具?安装一个快递批量查询高手...

3分钟查询物流,教你一键批量查询全部物流信息

很多朋友在问,如何在短时间内把单号的物流信息查询出来,查询完成后筛选已签收件、筛选未签收件,今天小编就分享一款物流查询神器,感兴趣的朋友接着往下看。第一步,运行【快递批量查询高手】在主界面中点击【添...

快递单号查询,一次性查询全部物流信息

现在各种快递的查询方式,各有各的好,各有各的劣,总的来说,还是有比较方便的。今天小编就给大家分享一个新的技巧,支持多家快递,一次能查询多个单号的物流,还能对查询到的物流进行分析、筛选以及导出,下面一起...

快递查询工具,批量查询多个快递快递单号的物流状态、签收时间

最近有朋友在问,怎么快速查询单号的物流信息呢?除了官网,还有没有更简单的方法呢?小编的回答当然是有的,下面一起来看看。需要哪些工具?安装一个快递批量查询高手多个京东的快递单号怎么快速查询?进入快递批量...

快递查询软件,自动识别查询快递单号查询方法

当你拥有多个快递单号的时候,该如何快速查询物流信息?比如单号没有快递公司时,又该如何自动识别再去查询呢?不知道如何操作的宝贝们,下面随小编一起来试试。需要哪些工具?安装一个快递批量查询高手快递单号若干...

教你怎样查询快递查询单号并保存物流信息

商家发货,快递揽收后,一般会直接手动复制到官网上一个个查询物流,那么久而久之,就会觉得查询变得特别繁琐,今天小编给大家分享一个新的技巧,下面一起来试试。教程之前,我们来预览一下用快递批量查询高手...

简单几步骤查询所有快递物流信息

在高峰期订单量大的时候,可能需要一双手当十双手去查询快递物流,但是由于逐一去查询,效率极低,追踪困难。那么今天小编给大家分享一个新的技巧,一次能查询多个快递单号的物流,下面一起来学习一下,希望能给大家...

物流单号查询,如何查询快递信息,按最后更新时间搜索需要的单号

最近有很多朋友在问,如何通过快递单号查询物流信息,并按最后更新时间搜索出需要的单号呢?下面随小编一起来试试吧。需要哪些工具?安装一个快递批量查询高手快递单号若干怎么快速查询?运行【快递批量查询高手】...

连续保存新单号功能解析,导入单号查询并自动识别批量查快递信息

快递查询已经成为我们日常生活中不可或缺的一部分。然而,面对海量的快递单号,如何高效、准确地查询每一个快递的物流信息,成为了许多人头疼的问题。幸运的是,随着科技的进步,一款名为“快递批量查询高手”的软件...

快递查询教程,快递单号查询,筛选更新量为1的单号

最近有很多朋友在问,怎么快速查询快递单号的物流,并筛选出更新量为1的单号呢?今天小编给大家分享一个新方法,一起来试试吧。需要哪些工具?安装一个快递批量查询高手多个快递单号怎么快速查询?运行【快递批量查...

掌握批量查询快递动态的技巧,一键查找无信息记录的两种方法解析

在快节奏的商业环境中,高效的物流查询是确保业务顺畅运行的关键。作为快递查询达人,我深知时间的宝贵,因此,今天我将向大家介绍一款强大的工具——快递批量查询高手软件。这款软件能够帮助你批量查询快递动态,一...

从复杂到简单的单号查询,一键清除单号中的符号并批量查快递信息

在繁忙的商务与日常生活中,快递查询已成为不可或缺的一环。然而,面对海量的单号,逐一查询不仅耗时费力,还容易出错。现在,有了快递批量查询高手软件,一切变得简单明了。只需一键,即可搞定单号查询,一键处理单...

物流单号查询,在哪里查询快递

如果在快递单号多的情况,你还在一个个复制粘贴到官网上手动查询,是一件非常麻烦的事情。于是乎今天小编给大家分享一个新的技巧,下面一起来试试。需要哪些工具?安装一个快递批量查询高手快递单号怎么快速查询?...

取消回复欢迎 发表评论: