Linux 信号量之Posix基于内存的信号量
信号量(semaphore),也和互斥锁一样提供了线程间或者进程间的同步功能。
信号量有三种:
- Posix有名字的信号量
- Posix基于内存的信号量
- System V信号量
信号量比互斥锁高级,互斥锁只允许一个线程访问临界区,信号量可以多个,可以把信号量看作成互斥锁的升级版,但是如果能用互斥锁解决,就用互斥锁,互斥锁比信号量节省资源。
这篇文章只介绍Posix基于内存的信号量
1,单个生产者和单个消费者
#include <pthread.h>
#include <stdlib.h>
#include <stdio.h>
#include <fcntl.h>
#include <sys/stat.h>
#include <semaphore.h>
#define NBUFF 10
int nitems;
struct {
int buff[NBUFF];
sem_t mutex, nempty, nstored;
} shared;
void* produce(void *args);
void* consume(void* args);
int main(int argc, char** argv){
pthread_t tid_produce, tid_consume;
if(argc != 2){
printf("usage error\n");
exit(1);
}
nitems = atoi(argv[1]);
//create 3 semaphore
sem_init(&shared.mutex, 0, 1);
sem_init(&shared.nempty, 0, NBUFF);
sem_init(&shared.nstored, 0, 0);
pthread_create(&tid_produce, NULL, produce, NULL);
pthread_create(&tid_consume, NULL, consume, NULL);
pthread_join(tid_produce, NULL);
pthread_join(tid_consume, NULL);
sem_destroy(&shared.mutex);
sem_destroy(&shared.nempty);
sem_destroy(&shared.nstored);
exit(0);
}
void* produce(void *args){
int i;
for(i = 0; i < nitems; ++i){
sem_wait(&shared.nempty);
sem_wait(&shared.mutex);
shared.buff[i % NBUFF] = i;
sem_post(&shared.mutex);
sem_post(&shared.nstored);
}
return NULL;
}
void* consume(void* args){
int i;
for(i = 0; i < nitems; ++i){
sem_wait(&shared.nstored);
sem_wait(&shared.mutex);
shared.buff[i % NBUFF] = i;
sem_post(&shared.mutex);
sem_post(&shared.nempty);
}
return NULL;
}
2,多个生产者和单个消费者
#include <pthread.h>
#include <stdlib.h>
#include <stdio.h>
#include <fcntl.h>
#include <sys/stat.h>
#include <semaphore.h>
#define NBUFF 10
#define MAXTHRS 100
#define min(x,y) ( x > y ? y:x )
int nitems, nproducers;
struct {
int buff[NBUFF];
int idx;
int val;
sem_t mutex, nempty, nstored;
} shared;
void* produce(void *args);
void* consume(void* args);
int main(int argc, char** argv){
int i, count[MAXTHRS];
pthread_t tid_produce[MAXTHRS], tid_consume;
if(argc != 3){
printf("usage error\n");
exit(1);
}
nitems = atoi(argv[1]);
nproducers = min(atoi(argv[2]), MAXTHRS);
//create 3 semaphore
sem_init(&shared.mutex, 0, 1);
sem_init(&shared.nempty, 0, NBUFF);
sem_init(&shared.nstored, 0, 0);
for(i = 0; i < nproducers; ++i){
count[i] = 0;
pthread_create(&tid_produce[i], NULL, produce, &count[i]);
}
pthread_create(&tid_consume, NULL, consume, NULL);
for(i = 0; i < nproducers; ++i){
pthread_join(tid_produce[i], NULL);
printf("count[%d] = %d\n", i, count[i]);
}
pthread_join(tid_consume, NULL);
sem_destroy(&shared.mutex);
sem_destroy(&shared.nempty);
sem_destroy(&shared.nstored);
exit(0);
}
void* produce(void *arg){
int i;
for(i = 0; i < nitems; ++i){
sem_wait(&shared.nempty);
sem_wait(&shared.mutex);
if(shared.idx >= nitems){
sem_post(&shared.nempty);//注意点
sem_post(&shared.mutex);
return NULL;// all done
}
shared.buff[shared.idx % NBUFF] = shared.val;
shared.idx++;
shared.val++;
sem_post(&shared.mutex);
sem_post(&shared.nstored);
*((int*) arg) += 1;
}
return NULL;
}
void* consume(void* args){
int i;
for(i = 0; i < nitems; ++i){
sem_wait(&shared.nstored);
sem_wait(&shared.mutex);
if(shared.buff[i % NBUFF] != i){
printf("error:buff[%d] = %d\n", i, shared.buff[i % NBUFF]);
}
sem_post(&shared.mutex);
sem_post(&shared.nempty);
}
return NULL;
}
3,多个生产者和多个消费者
#include <pthread.h>
#include <stdlib.h>
#include <stdio.h>
#include <fcntl.h>
#include <sys/stat.h>
#include <semaphore.h>
#define NBUFF 10
#define MAXTHRS 100
#define min(x,y) ( x > y ? y:x )
int nitems, nproducers, nconsumers;
struct {
int buff[NBUFF];
int idx;
int val;
int gidx;
int gval;
sem_t mutex, nempty, nstored;
} shared;
void* produce(void *args);
void* consume(void* args);
int main(int argc, char** argv){
int i, prodcount[MAXTHRS], conscount[MAXTHRS];
pthread_t tid_produce[MAXTHRS], tid_consume[MAXTHRS];
if(argc != 4){
printf("usage error\n");
exit(1);
}
nitems = atoi(argv[1]);
nproducers = min(atoi(argv[2]), MAXTHRS);
nconsumers = min(atoi(argv[3]), MAXTHRS);
//create 3 semaphore
sem_init(&shared.mutex, 0, 1);
sem_init(&shared.nempty, 0, NBUFF);
sem_init(&shared.nstored, 0, 0);
for(i = 0; i < nproducers; ++i){
prodcount[i] = 0;
pthread_create(&tid_produce[i], NULL, produce, &prodcount[i]);
}
for(i = 0; i < nconsumers; ++i){
conscount[i] = 0;
pthread_create(&tid_consume[i], NULL, consume, &conscount[i]);
}
for(i = 0; i < nproducers; ++i){
pthread_join(tid_produce[i], NULL);
printf("prodcount[%d] = %d\n", i, prodcount[i]);
}
for(i = 0; i < nconsumers; ++i){
pthread_join(tid_consume[i], NULL);
printf("conscount[%d] = %d\n", i, conscount[i]);
}
sem_destroy(&shared.mutex);
sem_destroy(&shared.nempty);
sem_destroy(&shared.nstored);
exit(0);
}
void* produce(void *arg){
int i;
for(i = 0; i < nitems; ++i){
sem_wait(&shared.nempty);
sem_wait(&shared.mutex);
if(shared.idx >= nitems){
sem_post(&shared.nstored);//注意点
sem_post(&shared.nempty);//注意点
sem_post(&shared.mutex);
return NULL;// all done
}
shared.buff[shared.idx % NBUFF] = shared.val;
shared.idx++;
shared.val++;
sem_post(&shared.mutex);
sem_post(&shared.nstored);
*((int*) arg) += 1;
}
return NULL;
}
void* consume(void* arg){
int i;
for(; ;){
sem_wait(&shared.nstored);
sem_wait(&shared.mutex);
if(shared.gidx >= nitems){
sem_post(&shared.nstored);//注意点
sem_post(&shared.mutex);
return NULL;// all done
}
i = shared.gidx % NBUFF;
if(shared.buff[i] != shared.gval){
printf("error:buff[%d] = %d\n", i, shared.buff[i]);
}
shared.gidx++;
shared.gval++;
sem_post(&shared.mutex);
sem_post(&shared.nempty);
*((int*) arg) += 1;
}
return NULL;
}
c/c++ 学习互助QQ群:877684253
本人微信:xiaoshitou5854
Linux 信号量之Posix基于内存的信号量的更多相关文章
- Linux 信号量之Posix有名字的信号量
信号量(semaphore),也和互斥锁一样提供了线程间或者进程间的同步功能. 信号量有三种: Posix有名字的信号量 Posix基于内存的信号量 System V信号量 信号量比互斥锁高级,互斥锁 ...
- linux第11天 共享内存和信号量
今天主要学习了共享内存和信号量 在此之前,有个管道问题 ls | grep a 整句话的意思是将ls输出到管道的写端,而流通到另一端的读端,grep a则是从管道的读端读取相关数据,再做筛选 共享内存 ...
- Linux进程同步之POSIX信号量
POSIX信号量是属于POSIX标准系统接口定义的实时扩展部分.在SUS(Single UNIX Specification)单一规范中,定义的XSI IPC中也同样定义了人们通常称为System V ...
- system V信号量和Posix信号量
一.函数上的区别 信号量有两种实现:传统的System V信号量和新的POSIX信号量.它们所提供的函数很容易被区分:对于所有System V信号量函数,在它们的名字里面没有下划线.例如,应该是sem ...
- 信号量(Posix)
Posix信号量分为有名信号量和无名信号量 1. Posix有名信号量 有名信号量既可以用于线程间的同步也可以用于进程间的同步 sem都是创建在/dev/shm目录下,名字格式sem.xxx,只需要指 ...
- 第三十三章 System V共享内存与信号量综合
用信号量解决生产者.消费者问题 实现shmfifo ip.h #ifndef _IPC_H #define _IPC_H #include <unistd.h> #include < ...
- linux网络编程-posix信号量与互斥锁(39)
-posix信号量信号量 是打开一个有名的信号量 sem_init是打开一个无名的信号量,无名信号量的销毁用sem_destroy sem_wait和sem_post是对信号量进行pv操作,既可以使用 ...
- Linux下用信号量实现对共享内存的访问保护
转自:http://www.cppblog.com/zjl-1026-2001/archive/2010/03/03/108768.html 最近一直在研究多进程间通过共享内存来实现通信的事情,以便高 ...
- Linux 内核同步之自旋锁与信号量的异同【转】
转自:http://blog.csdn.net/liuxd3000/article/details/8567070 Linux 设备驱动中必须解决的一个问题是多个进程对共享资源的并发访问,并发访问会导 ...
随机推荐
- [视频教程] ubuntu系统下安装最新版PHP7.3.X环境
视频地址: https://www.bilibili.com/video/av69088870/ 笔记: 先安装一下这个命令 add-apt-repositoryapt-get install sof ...
- 设备树处理之——device_node转换成platform_device【转】
转自:https://www.cnblogs.com/downey-blog/p/10486568.html 以下讨论基于linux4.14,arm平台 platform device 设备树的产生就 ...
- 第十章 Centos7-系统进程管理
第十章 Centos7-系统进程管理 本节所讲内容: 10.1 进程概述和ps查看进程工具 10.2 uptime查看系统负载-top动态管理进程 10.3 前后台进程切换- nice进程优先 ...
- 浅谈P/NP问题
克雷数学研究所(Clay Mathematics Institute,CMI)是在1998年由商人兰顿·克雷(Landon T. Clay)和哈佛大学数学家亚瑟·杰夫(Arthur Jaffe)创立, ...
- 一套从alpine基本镜像到node8.16.2的全套dockerfile
这个花了点时间,可以正式跑起来了. 加了常用的工具及中文时区,非root帐号. 除了pm2,其它的module放到应用程序本身的node_modules目录下来实现的. 一,3rd_part/node ...
- elementui-如何同时获取多选框的label和value
<!DOCTYPE html> <html> <head> <meta charset="UTF-8"> <!-- impor ...
- 【cf915】E. Physical Education Lessons(线段树)
传送门 简单的线段树区间修改区间查询,但是因为数据范围过大,所以采用动态开点的方法(注意一下空间问题). 也可以直接对询问区间的端点离散化然后建树,这种方法时间复杂度和空间复杂度都比较优秀. 给出动态 ...
- 10. Go语言—for循环
一.for循环 for 初始化语句;条件判断;变量修改 for i := 0; i < 100; i++{ fmt.Printf('i=%d\n',i) } // C 的 while 一样 fo ...
- lua 2 变量
变量在使用前,必须在代码中进行声明,即创建该变量. 编译程序执行代码之前编译器需要知道如何给语句变量开辟存储区,用于存储变量的值. Lua 变量有三种类型:全局变量.局部变量.表中的域. Lua 中的 ...
- 剑指Offer-16.合并两个排序的链表(C++/Java)
题目: 输入两个单调递增的链表,输出两个链表合成后的链表,当然我们需要合成后的链表满足单调不减规则. 分析: 可以用一个新的节点,来去比较两个单调递增的链表当前节点的值,如果p1当前的值小于p2,则新 ...