一起talk C栗子吧(第一百一十九回:C语言实例--线程死锁三)
来源:互联网 发布:域名续费怎么收费 编辑:程序博客网 时间:2024/03/29 13:26
各位看官们,大家好,上一回中咱们说的是线程死锁的例子,这一回咱们继续说该例子。闲话休提,言归正转。让我们一起talk C栗子吧!
看官们,由于篇幅的原因我们在上一回只介绍了死锁发生的第一种原因,今天我们将介绍死锁发生的第二种原因,并且该原因中的伪代码转换为实际的C语言代码。
为了方便,我们使用前面章回中演示互斥量的代码,在该代码的基础上做一些小修改来演示死锁。代码如下:
首先定义两个互斥量,互斥量是全局变量,方便线程使用。
#if MUTEX_ENABLEpthread_mutex_t mutex_value1;pthread_mutex_t mutex_value2;#endif
接下来在主进程中(也就是main函数)初始化两个互斥量:
res = pthread_mutex_init(&mutex_value1,NULL); res = pthread_mutex_init(&mutex_value2,NULL);
在主进程的最后还要记得释放与互斥量相关的资源:
#if MUTEX_ENABLE //destroy mutex res = pthread_mutex_destroy(&mutex_value1); res = pthread_mutex_destroy(&mutex_value2);#endif
我们分别修改两个线程的执行函数,该段代码是核心代码,请大家仔细阅读:
// the first thread functionvoid *thread_func1(void *param){ int i = 0; int res = 0; pthread_t thread_id; thread_id = pthread_self(); printf("Thread ID::%u -----------S---------- \n",(unsigned int)thread_id); while(i++ < 4) {#if MUTEX_ENABLE res = pthread_mutex_lock(&mutex_value1); // mutex1 is locking if(res != 0) { printf(" mutex1 lock failed \n"); }#endif read_data("Thread_1");#if MUTEX_ENABLE res = pthread_mutex_lock(&mutex_value2); //mutex2 is locking if(res != 0) { printf(" mutex2 lock failed \n"); }#endif#if MUTEX_ENABLE res = pthread_mutex_unlock(&mutex_value2); if(res != 0) { printf(" mutex2 unlock failed \n"); } res = pthread_mutex_unlock(&mutex_value1); if(res != 0) { printf(" mutex1 unlock failed \n"); }#endif sleep(2); } printf("Thread ID::%u -----------E---------- \n",(unsigned int)thread_id); pthread_exit(&status); // end the thread}
// the second thread functionvoid *thread_func2(void *param){ int i = 0; int res = 0; pthread_t thread_id; thread_id = pthread_self(); printf("Thread ID::%u -----------S---------- \n",(unsigned int)thread_id); while(i++ < 4) {#if MUTEX_ENABLE res = pthread_mutex_lock(&mutex_value2); //mutex 2 is locking if(res != 0) { printf(" mutex2 lock failed \n"); }#endif write_data("Thread_2");#if MUTEX_ENABLE res = pthread_mutex_lock(&mutex_value1); //mutex 1 is locking if(res != 0) { printf(" mutex1 lock failed \n"); }#endif#if MUTEX_ENABLE res = pthread_mutex_unlock(&mutex_value1); if(res != 0) { printf(" mutex1 unlock failed \n"); }#endif#if MUTEX_ENABLE res = pthread_mutex_unlock(&mutex_value2); if(res != 0) { printf(" mutex2 unlock failed \n"); }#endif sleep(1); } printf("Thread ID::%u -----------E---------- \n",(unsigned int)thread_id); pthread_exit(&status); // end the thread}
我们运行上面的程序,可以得到以下结果:
Create first thread //创建第一个线程Create second thread //创建第二个线程Thread ID::3076344640 -----------S---------- [Thread_1] start reading data //第一个线程读取数据,同时对互斥量一加锁Thread ID::3067951936 -----------S---------- [Thread_2] start writing data //第二个线程修改数据,同时对互斥量二加锁[Thread_1] data = 0 [Thread_1] end reading data //第一个线程读取数据结束,同时等待互斥量二被解锁[Thread_2] data = 1 [Thread_2] end writing data //第二个线程修改数据结束,同时等待互斥量一被解锁mutex2 can't be destroyed //发生死锁,程序运行错误
从上面的程序运行结果中可以看到,线程1锁住了互斥量一,同时等待互斥量二;而线程2锁住了互斥量二,同时等待互斥量一。这样便造成了死锁,进而引起了程序运行错误。
该程序是为了演示死锁的原因专门写的,这样写程序不合理,因为它不能同步线程,大家不要拘泥于程序的内容,重点是理解死锁是如何发生的。
看官们,正文中就不写代码了,完成的代码放到了我的资源中,大家可以点击这里下载使用。
各位看官,关于线程死锁的例子咱们就说到这里。欲知后面还有什么例子,且听下回分解 。
1 0
- 一起talk C栗子吧(第一百一十九回:C语言实例--线程死锁三)
- 一起talk C栗子吧(第一百一十七回:C语言实例--线程死锁一)
- 一起talk C栗子吧(第一百一十八回:C语言实例--线程死锁二)
- 一起talk C栗子吧(第一百一十回:C语言实例--线程标识符)
- 一起talk C栗子吧(第一百一十一回:C语言实例--线程间通信)
- 一起talk C栗子吧(第一百一十二回:C语言实例--线程同步概述)
- 一起talk C栗子吧(第一百一十三回:C语言实例--线程同步之信号量一)
- 一起talk C栗子吧(第一百一十四回:C语言实例--线程同步之信号量二)
- 一起talk C栗子吧(第一百一十五回:C语言实例--线程同步之互斥量一)
- 一起talk C栗子吧(第一百一十六回:C语言实例--线程同步之互斥量二)
- 一起talk C栗子吧(第一百二十回:C语言实例--线程属性)
- 一起talk C栗子吧(第一百二十九回:C语言实例--C程序内存布局一)
- 一起talk C栗子吧(第二十九回:C语言实例--选择排序)
- 一起talk C栗子吧(第四十九回:C语言实例--最小生成树一)
- 一起talk C栗子吧(第五十九回:C语言实例--字符串概述)
- 一起talk C栗子吧(第六十九回:C语言实例--DIY字符串查找函数)
- 一起talk C栗子吧(第一百二十四回:C语言实例--内置宏)
- 一起talk C栗子吧(第一百二十六回:C语言实例--static关键字)
- 弱语言变量作用域
- Maven主要插件汇总
- 结对编程的正确姿势,你会了吗?
- iOS生命周期
- 关于解决ubuntu中Docky软件断电自动关闭的问题(Linux: Docky Closes After Waking Up From Suspend (Ubuntu 14.04))
- 一起talk C栗子吧(第一百一十九回:C语言实例--线程死锁三)
- HDOJ 1217 Arbitrage (最短路 Floyd)
- Gridpanel符合条件的行有颜色
- python程序编译成exe文件
- libmad编译移植到A20
- P29 (**) Compose arbitrarily many procedures of one argument.
- 警告: A docBase inside the host appBase has been specified, and will be ignore2
- dmesg时间转换工具
- ubuntu 寝室拨号上网以及在家上网