linux系统编程:线程同步-信号量(semaphore)

news/2024/7/8 4:45:08 标签: semaphore, 线程, 信号量, 线程同步

                             线程同步-信号量(semaphore)

生产者与消费者问题再思考

在实际生活中,只要有商品,消费者就可以消费,这没问题。但生产者的生产并不是无限的,例如,仓库是有限的,原材料是有限的,生产指标受消费指标限制等等。为了进一步,解决好生产者与消费者问题,引入信号量进机制。


信号量

信号量(semaphore)是互斥量的升级版:互斥量的状态为0或1,而信号量可以为n。也就是说,使用互斥量时,最多允许一个线程进入关键区,而信号量允许多个,具体值是信号量当前的内部值。


相关函数

sem_t       //信号量类型
sem_init(sem_t *sem, int pshared, unsigned int value);
sem_wait(sem_t *sem)
sem_trywait
sem_timedwait
sem_post(sem_t *sem)
sem_destroy
重要的是理解:sem_wait和sem_post两个函数。

sem_wait(sem);当sem为零时,线程阻塞;否则,sem减一,线程不阻塞。

sem_post(sem);sem加一。

此外,使用sem_init方法,对信号量类型初始化,第二个参数,默认是0,标明用于线程之间。第三个参数指定了初始值。


单生产者与单消费者

#include <stdio.h>
#include <unistd.h>
#include <pthread.h>
#include <semaphore.h>
#define NUM 5
sem_t blank_num, product_num;
int i, j, k;
int goods[NUM];
void *producer(void *argv)
{
	while (1)
	{
		sem_wait(&blank_num);
		goods[i] = rand() % 100 + 1;
		printf("produce %d\n", goods[i]);
		sem_post(&product_num);
		i = (i + 1) % NUM;
		sleep(rand() % 2);
	}
}
void *comsumer(void *argv)
{
	while (1)
	{
		sem_wait(&product_num);
		printf("comsume %d\n", goods[j]);
		goods[j] = 0;
		sem_post(&blank_num);
		j = (j + 1) % NUM;
		sleep(rand() % 2);
	}
}
int main(void)
{
	i = j = k = 0;
	//初始化信号量
	sem_init(&blank_num, 0, NUM);
	sem_init(&product_num, 0, 0);
	pthread_t pro, com;
	pthread_create(&com, NULL, producer, NULL);
	pthread_create(&pro, NULL, comsumer, NULL);
	pthread_join(com, NULL);
	pthread_join(pro, NULL);
	sem_destroy(&blank_num);
	sem_destroy(&product_num);
	return 0;
}


多生产者与多消费者

#include <stdio.h>
#include <unistd.h>
#include <pthread.h>
#include <semaphore.h>
#define NUM 5
pthread_mutex_t m1, m2;
sem_t blank_num, product_num;
int goods[NUM];
int i, j, k;
void *producer(void *argv)
{
	while (1)
	{
		sem_wait(&blank_num);
		pthread_mutex_lock(&m1);
		goods[i] = rand() % 100 + 1;
		printf("produce %d\n", goods[i]);
		i = (i + 1) % NUM;
		pthread_mutex_unlock(&m1);
		sem_post(&product_num);
		sleep(rand() % 2);
	}
}
void *comsumer(void *argv)
{
	while (1)
	{
		sem_wait(&product_num);
		pthread_mutex_lock(&m2);
		printf("comsume %d\n", goods[j]);
		goods[j] = 0;   //置零
		j = (j + 1) % NUM;
		pthread_mutex_unlock(&m2);
		sem_post(&blank_num);
		sleep(rand() % 2);
	}
}
int main(void)
{
	i = j = k = 0;
	//初始化信号量及互斥量
	sem_init(&blank_num, 0, NUM);
	sem_init(&product_num, 0, 0);
	pthread_mutex_init(&m1, NULL);
	pthread_mutex_init(&m2, NULL);
	pthread_t pro[2], com[3];
	for (k = 0; k < 3; k++)
		pthread_create(&com[k], NULL, producer, NULL);
	for (k = 0; k < 2; k++)
		pthread_create(&pro[k], NULL, comsumer, NULL);
	for (k = 0; k < 3; k++)
		pthread_join(com[k], NULL);
	for (k = 0; k < 2; k++)
		pthread_join(pro[k], NULL);
	pthread_mutex_destroy(&m1);
	pthread_mutex_destroy(&m2);
	sem_destroy(&blank_num);
	sem_destroy(&product_num);
	return 0;
}


     

CCPP Blog 目录



http://www.niftyadmin.cn/n/1802091.html

相关文章

挑战面试编程:查找数组中第k大的数

查找数组中第k大的数 问题&#xff1a; 查找出一给定数组中第k大的数。例如[3,2,7,1,8,9,6,5,4]&#xff0c;第1大的数是9&#xff0c;第2大的数是8…… 思路&#xff1a; 1. 直接从大到小排序&#xff0c;排好序后&#xff0c;第k大的数就是arr[k-1]。 2. 只需找到第k大的…

CCPP Blog 目录

CCPP Blog 目录 专栏目录 数据结构与算法C指针C拾遗挑战面试编程 十六进制数转化为八进制数链表逆转的多种实现字符串匹配的双重递归式写法原码、反码、补码字符串替换字符串包含回文串、回文数字单词翻转、高斯公式、魔方矩阵、黑白球、3n1最大连续子序列和字符串转换为整数计…

C语言运行顺序和内存就地分配原则

不知道&#xff0c;为什么会出现这样的结果&#xff1f;答&#xff1a;此问题从二个方面解答&#xff1a;1> 编译器是上古三大神器的编译器之一&#xff0c;不带有任何只能辅助系统。2>本程序在语法没有任何问题&#xff0c;但是根据C语言编译语句自上而下 和 内存就地分…

编程中的问题

int arr[n]{1,2,3,4,5,6,7,8,9,11}?这个 N的值 是如何传来滴&#xff01;

vi 與 vim 的指令整理

&#xfeff;&#xfeff;vi 與 vim 的指令整理转自&#xff1a;http://www.vixual.net/blog/archives/tag/linuxvi 是 unix 家族下最功能強大的文字編輯器&#xff0c;讓用戶只要使用一個鍵盤就可以完成所有的編輯。而 vim 則是 vi 的加強版&#xff0c;甚至在 Windows 上也找…

在Windows 10下搭建Ubuntu Linux+GCC开发平台

&#xfeff;&#xfeff;在Windows 10下搭建Ubuntu LinuxGCC开发平台 2017年06月02日 14:40:11 1351人阅读 评论(1) 收藏 举报 目录(?)[-]第一步启用Linux子系统功能第二步启用开发人员模式第三步下载Linux子系统此步涉及到过程中最最最大的难点耐性第四步进入子系统第五…

Ubuntu下gcc安装及使用

&#xfeff;&#xfeff;Ubuntu下gcc安装及使用 2016年06月21日 10:58:37 29486人阅读 评论(2) 收藏 举报 分类&#xff1a; Ubuntu&#xff08;6&#xff09; 作者同类文章 X 目录(?)[]一安装二编译在Ubuntu下安装GCC和其他一些Linux系统有点不一样。一、安装方法一&a…

unix文件和目录【转载】

&#xfeff;&#xfeff;unix文件和目录原创 2015年07月12日 15:04:41 标签&#xff1a;unix /文件系统 /操作系统 /805编辑 删除 文件系统是存储数据的基础&#xff0c;对于一个操作系统来说至关重要&#xff0c;unix支持多种文件系统&#xff0c;各文件系统的特性有所不同…