【linux】线程同步

互斥锁

主要应用函数

  • pthread_mutex_init

    • 静态初始化:如果互斥锁 mutex 是静态分配的(定义在全局,或加了static关键字修饰),可以直接使用宏进行初始化。 pthead_mutex_t muetx = PTHREAD_MUTEX_INITIALIZER;

    • 动态初始化:局部变量应采用动态初始化。pthread_mutex_init(&mutex, NULL)

  • pthread_mutex_lock
    函数加锁。可理解为将mutex–(或-1)
    int pthread_mutex_lock(pthread_mutex_t *mutex);

  • pthread_mutex_unlock
    函数解锁。可理解为将mutex ++(或+1)
    int pthread_mutex_unlock(pthread_mutex_t *mutex);

  • pthread_mutex_t 类型主要用于定义互斥量

读写锁

  • 允许更高的并行性
  • 写独占,读共享
  • 写比读优先

主要应用函数

  • pthread_rwlock_init函数
  • pthread_rwlock_destroy函数
  • pthread_rwlock_rdlock函数
  • pthread_rwlock_wrlock函数
  • pthread_rwlock_tryrdlock函数
  • pthread_rwlock_trywrlock函数
  • pthread_rwlock_unlock函数

条件变量

条件变量本身不是锁。但它也可以造成线程阻塞。通常与互斥锁配合使用。

主要应用函数

  • pthread_cond_init函数

  • pthread_cond_destroy函数

  • pthread_cond_wait函数阻塞等待一个条件变量
    int pthread_cond_wait(pthread_cond_t *restrict cond, pthread_mutex_t *restrict mutex);

    • 阻塞等待条件变量cond(参1)满足
    • 释放已掌握的互斥锁(解锁互斥量)相当于pthread_mutex_unlock(&mutex);
    • 当被唤醒,pthread_cond_wait函数返回时,解除阻塞并重新申请获取互斥锁pthread_mutex_lock(&mutex);
  • pthread_cond_timedwait函数

  • pthread_cond_signal函数:唤醒至少一个阻塞在条件变量上的线程

  • pthread_cond_broadcast函数:唤醒全部阻塞在条件变量上的线程

  • pthread_cond_t类型 用于定义条件变量

生产者消费者问题代码

#include <pthread.h>
#include <stdlib.h>
#include <stdio.h>
#include <unistd.h>

struct msg { //利用链表存储,一个msg代表一个节点
	int val;
	msg* next;
};

struct msg *head, *mp;

pthread_cond_t has_product = PTHREAD_COND_INITIALIZER; //init
pthread_mutex_t lock = PTHREAD_MUTEX_INITIALIZER; //init

void *producer(void *arg) {
	while(1) {
		mp = (struct msg*)malloc(sizeof(struct msg));
		mp->val = rand() % 1000 + 1; // 产生随机数,代表生产一个产品的代号
		printf("producer:%d\n", mp->val);
		
		pthread_mutex_lock(&lock); //上锁
		mp->next = head; //利用头插法将节点插入链表
		head = mp; //
		pthread_mutex_unlock(&lock);//解锁
		
		pthread_cond_signal(&has_product); //唤醒阻塞在has_product条件变量上的线程
		
		sleep(rand() % 3);
		
	}
	
}

void *consumer(void *arg) {
	while(1) {
		pthread_mutex_lock(&lock); // 上锁
		while(head == NULL) {  // 当没有产品时,阻塞条件变量,并且解锁,这两步是原子操作。注意这里使用while而不是if,防止多个消费者共同抢占资源的情况
			pthread_cond_wait(&has_product, &lock); 
		}
		
		mp = head;//将头结点删去
		head = head->next; 
		pthread_mutex_unlock(&lock); //解锁
		
		printf("consumer:%d\n", mp->val);
		free(mp); //释放节点资源
		sleep(rand() %  3);
	}
	
}

int main() {
	pthread_t pid, cid;
	srand(time(NULL)); //随机数种子
	
	pthread_create(&pid, NULL, producer, NULL); //创建子线程
	pthread_create(&cid, NULL, consumer, NULL);
	
	pthread_join(pid, NULL); //释放子线程
	pthread_join(cid, NULL);
	
	return 0;
}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值