介绍
“真诚服务,让网络创造价值”是我们的服务理念,创新互联团队十余年如一日始终坚持在网站建设领域,为客户提供优质服。不管你处于什么行业,助你轻松跨入“互联网+”时代,PC网站+手机网站+公众号+微信小程序定制开发。
Linux操作系统中,信号量是用于进程或线程之间通信和互斥控制的一种机制。信号量可用于协调多个进程或线程之间的访问。在多个进程或线程之间,通过使用信号量机制,可以确保只有一个进程或线程可以访问共享资源。
信号量分为两种:二元信号量和计数信号量。其中,二元信号量只能取0或1两个值,适合用于互斥控制;计数信号量可以取任何非负整数,适合用于进程或线程之间通信。
本文将详细介绍Linux中信号量的使用方法以及在实践中如何使用信号量实现互斥控制。
一、信号量API
在Linux中,信号量的API有三个函数,分别是semget、semop和semctl。
1.1 semget
semget函数用于创建一个新的信号量或获取一个已存在的信号量。
基本的函数格式如下:
“`
#include
#include
#include
int semget(key_t key, int nsems, int sem);
“`
其中,key是信号量标识符,在创建或获取时需要指定一个唯一的标识符。nsems指定需要创建或获取的信号量数量,sem参数则是控制信号量的标志,通常使用IPC_CREAT标志创建一个新的信号量。
1.2 semop
semop函数用于对信号量进行P操作和V操作。
基本的函数格式如下:
“`
#include
#include
#include
int semop(int semid, struct sembuf *sops, size_t nsops);
“`
其中,semid是信号量的标识符,sops是一个指向sembuf结构数组的指针,每一个sembuf结构体表示要进行的操作(P或V)以及操作的信号量编号和操作数量。
1.3 semctl
semctl函数用于控制信号量的各种属性。
基本的函数格式如下:
“`
#include
#include
#include
int semctl(int semid, int semnum, int cmd, …);
“`
其中,semid是信号量的标识符,semnum是信号量的序号,cmd是要执行的操作码,后面的可选参数依赖于cmd的不同。
二、信号量的实际应用
信号量在操作系统中被广泛应用于进程或线程之间的通信和互斥控制,下面将详细介绍信号量在互斥控制中的应用。
2.1 互斥控制
在多进程或多线程程序中,为了确保在某个资源被使用时只有一个进程或线程能够访问该资源,需要使用信号量机制来实现互斥控制。
例如,假设在一个多进程程序中多个进程需要同时访问一个共享资源,为了避免多个进程同时访问该资源而引发的冲突,需要使用信号量来进行互斥控制。
以下是使用信号量控制互斥访问共享资源的示例程序:
“`
#include
#include
#include
#include
#include
#define SEM_KEY 39999
int semid;
int init_sem(int sem_key)
{
union semun arg;
int semid = semget(sem_key, 1, IPC_CREAT | IPC_EXCL | 0666);
if(semid == -1){
semid = semget(sem_key, 1, 0666);
if(semid == -1){
perror(“semget”);
exit(1);
}
}
arg.val = 1;
semctl(semid, 0, SETVAL, arg);
return semid;
}
void P(int semid)
{
struct sembuf sem_p;
sem_p.sem_num = 0;
sem_p.sem_op = -1;
sem_p.sem_ = SEM_UNDO;
semop(semid, &sem_p, 1);
}
void V(int semid)
{
struct sembuf sem_v;
sem_v.sem_num = 0;
sem_v.sem_op = 1;
sem_v.sem_ = SEM_UNDO;
semop(semid, &sem_v, 1);
}
int mn()
{
semid = init_sem(SEM_KEY);
pid_t pid;
pid = fork();
if(pid == -1){
perror(“fork”);
exit(1);
}else if(pid == 0){
P(semid);
printf(“Child process get the semaphore.\n”);
sleep(2);
V(semid);
printf(“Child process release the semaphore.\n”);
}else{
printf(“Parent process trying to get the semaphore.\n”);
P(semid);
printf(“Parent process get the semaphore.\n”);
V(semid);
printf(“Parent process release the semaphore.\n”);
}
return 0;
}
“`
在上述代码中,我们使用了三个函数P、V和init_sem来实现对共享资源的互斥访问。
其中,init_sem函数用于创建或获取一个信号量,并设定初始的值为1,即初始时资源未被占用。
P函数用于对共享资源进行加锁操作,即申请信号量。在申请信号量时,我们将信号量的值减1。如果信号量的值为0,则表示资源正在被占用,进程或线程需要等待。
V函数用于对共享资源进行解锁操作,即释放信号量。在释放信号量时,我们将信号量的值加1。如果有进程或线程在等待该资源,我们会将其唤醒并继续执行。
在上述代码中,我们首先创建了一个信号量,并将其值设为1。然后创建了一个子进程,在子进程中调用P函数占用信号量,表示子进程获得了对该资源的访问权。在P函数中,如果信号量的值为0,则表示资源正在被占用,进程或线程需要等待。在子进程等待了2秒钟后,使用V函数释放信号量,表示占用该资源的操作已经完成。
在子进程完成操作后,父进程尝试获取信号量,如果获取到了信号量,则表示父进程获得了对该资源的访问权。在父进程中,我们同样使用V函数释放占用的信号量。
通过上述互斥控制的示例程序,我们可以了解到,在编写多进程或多线程的程序时,使用信号量机制可以很好地控制资源的访问,保证操作的正确性和安全性。
3.
相关问题拓展阅读:
mutex保护的资源在同一时刻只允许一个task进行访问;semaphore根据初始值n可以允许至旁衡键多n个task访问。
semaphore可以实现拦咐“等待”机制,一种常见的场景是task0进入阻塞状态“等待”某个事件发生,task1触发事件后“唤醒”task0。task0在“等待”时处于阻塞运巧状态而不是运行状态,因此不会浪费CPU时间。而一个task在拿到mutex之后释放之前不宜进行太长时间的操作,更不能阻塞。
mutex互斥体只用于保护临界区的代码(访问共享资源),而桐袭漏不用于锁之间的同步,即一个线程释放mutex锁后,马上又可能获取同一个锁,禅镇而不管其它正在等待该mutex锁的其它线程。
semaphore信号量除了起到保护临界区的作用外,还用于锁同步的功能,即一个线程释放semaphore后,会保证正在等待该semaphore的线程优先执行,而不会马上在获取同一个semaphore。
如果两个局烂线程想通过一个锁达到输出1,2,1,2,1,2这样的序列,应使用semaphore, 而使用mutex的结果可能为1,1,1,1,1,2,2,2,111…..。
mutex互斥体只用于保护临界区的代码(访问共享资源),而桐袭漏不用于锁之间的同步,即一个线程释放mutex锁后,马上又可能获取同一个锁,禅镇而不管其它正在等待该mutex锁的其它线程。
semaphore信号量除了起到保护临界区的作用外,还用于锁同步的功能,即一个线程释放semaphore后,会保证正在等待该semaphore的线程优先执行,而不会马上在获取同一个semaphore。
如果两个局烂线程想通过一个锁达到输出1,2,1,2,1,2这样的序列,应使用semaphore, 而使用mutex的结果可能为1,1,1,1,1,2,2,2,111…..。
关于linux 信号量互斥控制的介绍到此就结束了,不知道你从中找到你需要的信息了吗 ?如果你还想了解更多这方面的信息,记得收藏关注本站。
香港服务器选创新互联,2H2G首月10元开通。
创新互联(www.cdcxhl.com)互联网服务提供商,拥有超过10年的服务器租用、服务器托管、云服务器、虚拟主机、网站系统开发经验。专业提供云主机、虚拟主机、域名注册、VPS主机、云服务器、香港云服务器、免备案服务器等。
分享标题:Linux中信号量互斥控制详解(linux信号量互斥控制)
标题链接:http://www.mswzjz.cn/qtweb/news1/523551.html
攀枝花网站建设、攀枝花网站运维推广公司-贝锐智能,是专注品牌与效果的网络营销公司;服务项目有等
声明:本网站发布的内容(图片、视频和文字)以用户投稿、用户转载内容为主,如果涉及侵权请尽快告知,我们将会在第一时间删除。文章观点不代表本网站立场,如需处理请联系客服。电话:028-86922220;邮箱:631063699@qq.com。内容未经允许不得转载,或转载时需注明来源: 贝锐智能