多线程时需要注意互斥锁的加锁顺序,防止死锁

多线程时需要注意互斥锁的加锁顺序,否则可能导致死锁,一个简单的例子:

static pthread_mutex_t A, B; 
void *printmsg1(void *msg) 
{ 
	// 加锁顺序为A、B
	pthread_mutex_lock(&A); 
	pthread_mutex_lock(&B);
	printf("printmsg1\n"); 
	pthread_mutex_unlock(&B); 
	pthread_mutex_unlock(&A); return 0; 
} 
void *printmsg2(void *msg) { 
	// 加锁顺序为B、A
	pthread_mutex_lock(&B); 
	pthread_mutex_lock(&A); 
	printf("printmsg2\n"); 
	pthread_mutex_unlock(&A); 
	pthread_mutex_unlock(&B); 
	return 0; 
} 
int main(int argc, char**argv) { 
	pthread_t pt1, pt2; 
	pthread_mutex_init(&A, NULL); 
	pthread_mutex_init(&B, NULL); 
	pthread_create(&pt1,0, printmsg1, NULL); 
	pthread_create(&pt2,0, printmsg2, NULL); 
	pthread_join(pt1,0); 
	pthread_join(pt2,0); 
	pthread_mutex_destroy(&A); 
	pthread_mutex_destroy(&B); 
	return 0; 
}

在上面这段代码中,线程pt1调用函数printmsg1,线程pt2调用函数printmsg2,有这两个函数中加锁的顺序不一致,就会很容易导致死锁。
假如某个状态,线程pt1调用函数printmsg1,获取了互斥锁A,线程pt2调用函数printmsg2获取了互斥锁B,由于线程pt1无法继续获取锁B,而处于阻塞,同样,线程pt2由于无法获取互斥锁A而进入阻塞,这样两个都阻塞了就会进入死锁。如果两个函数都按照A、B的顺序获取就不会导致这个问题。

同样,信号量也需要特别注意死锁问题:
先说明一下pthread_cond_wait函数:
该函数第一个参数为条件变量指针,第二个为互斥量指针。该函数调用前,需本线程加锁互斥量,加锁状态的时间内函数完成线程加入等待队列操作 ,线程进入等待前函数解锁互斥量。在满足条件离开pthread_cond_wait函数之前重新获得互斥量并加锁,因此,本线程之后需要再次解锁互斥量。

static pthread_mutex_t A, B; 
 static pthread_cond_t cond; 
 static int count; 
 void *f1(void *msg) 
 { 
	pthread_mutex_lock(&A); 
	pthread_mutex_lock(&B); 
	while (count) 
	{ 
		pthread_cond_wait(&cond, &B); 
	} 
	pthread_mutex_unlock(&B); 
	pthread_mutex_unlock(&A); 
	return 0; 
 } 
 
 void *f2(void *msg) 
 { 
	pthread_mutex_lock(&A); 
	pthread_mutex_lock(&B); 
	count--;
	pthread_cond_broadcast(&cond);
	pthread_mutex_unlock(&B);
	pthread_mutex_unlock(&A); 
	return 0; 
} 
int main(int argc, char**argv) 
 { 
	pthread_t pt1, pt2; 
	pthread_mutex_init(&A, NULL);
	pthread_mutex_init(&B, NULL); 
	pthread_cond_init(&cond, NULL); 
	count = 1; 
	pthread_create(&pt1,0, f1, NULL); 
	pthread_create(&pt2,0, f2, NULL); 
	pthread_join(pt1, NULL); 
	pthread_join(pt2, NULL); 
	pthread_mutex_destroy(&A); 
	pthread_mutex_destroy(&B);
	pthread_cond_destroy(&cond); return 0;
}

可能出现这样一种情况:
f1由于count =1从而执行函数:
pthread_cond_wait(&cond, &B);
此时虽然会释放了互斥锁B,之后进入阻塞状态,这里需要注意:此时互斥锁A还没有被释放。
而f2执行需要先获取互斥锁A以及锁B,但是虽然B已经被f1释放了,但是A仍然被f1占着,因此f2所在的线程也会阻塞,这样两个线程同时阻塞,就会导致互等的死锁状态。

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
互斥是一种保护共享资源的机制,多个线程可以通过互斥来访问共享资源,避免多个线程同时访问共享资源导致的竞争问题。在使用互斥时,需要注意以下几点: 1. 初始化互斥:在使用互斥前,需要先进行初始化。可以使用 `pthread_mutex_init()` 函数来进行初始化。 2. 和解:在访问共享资源前,需要。使用 `pthread_mutex_lock()` 函数来,使用 `pthread_mutex_unlock()` 函数来解后,其他线程无法访问共享资源,直到当前线程解。 3. 避免:由于互斥是一种独占资源的机制,如果多个线程同时请求多个互斥,很容易导致。为了避免需要对多个互斥顺序进行协调,保证所有线程按照相同的顺序请求互斥。 下面是一个使用互斥的简单示例代码: ```c++ #include <pthread.h> #include <stdio.h> pthread_mutex_t mutex; void *thread_func(void *) { pthread_mutex_lock(&mutex); // printf("Thread %ld is running\n", pthread_self()); pthread_mutex_unlock(&mutex); // 解 return NULL; } int main() { pthread_t threads[10]; pthread_mutex_init(&mutex, NULL); // 初始化互斥 for (int i = 0; i < 10; i++) { pthread_create(&threads[i], NULL, thread_func, NULL); } for (int i = 0; i < 10; i++) { pthread_join(threads[i], NULL); } pthread_mutex_destroy(&mutex); // 销毁互斥 return 0; } ``` 在上面的代码中,我们创建了 10 个线程,每个线程都会尝试获取互斥,打印自己的线程 ID,然后释放互斥。由于互斥的存在,每次只有一个线程能够获取到互斥,其他线需要等待前一个线程释放互斥后才能获取到互斥

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值