一起talk C栗子吧(第一百一十九回:C语言实例--线程死锁三)
各位看官们。大家好,上一回中咱们说的是线程死锁的样例,这一回咱们继续说该样例。闲话休提,言归正转。让我们一起talk C栗子吧!
看官们,由于篇幅的原因我们在上一回仅仅介绍了死锁发生的第一种原因,今天我们将介绍死锁发生的另外一种原因,而且该原因中的伪代码转换为实际的C语言代码。
为了方便,我们使用前面章回中演示相互排斥量的代码,在该代码的基础上做一些小改动来演示死锁。代码例如以下:
首先定义两个相互排斥量,相互排斥量是全局变量。方便线程使用。
#if MUTEX_ENABLE
pthread_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 function
void *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 function
void *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锁住了相互排斥量二,同一时候等待相互排斥量一。这样便造成了死锁,进而引起了程序执行错误。
该程序是为了演示死锁的原因专门写的,这样敲代码不合理。由于它不能同步线程。大家不要拘泥于程序的内容,重点是理解死锁是怎样发生的。
看官们,正文中就不写代码了,完毕的代码放到了我的资源中,大家能够点击这里下载使用。
各位看官,关于线程死锁的样例咱们就讲到这里。
欲知后面还有什么样例,且听下回分解 。
一起talk C栗子吧(第一百一十九回:C语言实例--线程死锁三)的更多相关文章
- 一起talk C栗子吧(第一百一十二回:C语言实例--线程同步概述)
各位看官们,大家好.上一回中咱们说的是线程间通信的样例,这一回咱们说的样例是:线程同步.闲话休提,言归正转.让我们一起talk C栗子吧! 看官们,提到同步.我想大家都不陌生,由于我们在前面章回中介绍 ...
- 一起talk C栗子吧(第十二回:C语言实例--单链表一)
各位看官们,大家好.从今天開始,我们讲大型章回体科技小说 :C栗子.也就是C语言实例.闲话休提, 言归正转. 让我们一起talk C栗子吧! 看官们,上一回中咱们没有说详细的样例,并且是说了样例中的文 ...
- 第一百一十九节,JavaScript事件入门
JavaScript事件入门 学习要点: 1.事件介绍 2.内联模型 3.脚本模型 4.事件处理函数 JavaScript事件是由访问Web页面的用户引起的一系列操作,例如:用户点击.当用户执行某些操 ...
- 一起talk C栗子吧(第二十二回:C语言实例--队列一)
各位看官们,大家好,上一回中咱们说的是表达式求值的样例,该样例使用了栈,这一回咱们说的是栈的 兄弟:队列. 闲话休提,言归正转.让我们一起talk C栗子吧. 我们在这里说的队列是一种抽象的数据结构, ...
- 一起talk C栗子吧(第一百二十一回:C语言实例--线程知识体系图)
各位看官们.大家好,上一回中咱们说的线程属性的样例.这一回咱们说的样例是:线程知识体系图.闲话休提.言归正转. 让我们一起talk C栗子吧! 我们在前面的章回中介绍了与线程相关的知识,在今天的章回中 ...
- “全栈2019”Java第一百一十二章:什么是闭包?
难度 初级 学习时间 10分钟 适合人群 零基础 开发语言 Java 开发环境 JDK v11 IntelliJ IDEA v2018.3 文章原文链接 "全栈2019"Java第 ...
- 第三百一十九节,Django框架,文件上传
第三百一十九节,Django框架,文件上传 1.自定义上传[推荐] 请求对象.FILES.get()获取上传文件的对象上传对象.name获取上传文件名称上传对象.chunks()获取上传数据包,字节码 ...
- [你必须知道的.NET]第十九回:对象创建始末(下)
本文将介绍以下内容: 对象的创建过程 内存分配分析 内存布局研究 接上回[第十八回:对象创建始末(上)],继续对对象创建话题的讨论>>> 2.2 托管堆的内存分配机制 引用类型的实例 ...
- 第一百一十六节,JavaScript,DOM操作样式
JavaScript,DOM操作样式 一.操作样式 CSS作为(X)HTML的辅助,可以增强页面的显示效果.但不是每个浏览器都能支持最新的CSS能力.CSS的能力和DOM级别密切相关,所以我们有必要检 ...
随机推荐
- (转)Android开发书籍推荐:从入门到精通系列学习路线书籍介绍
Android开发书籍推荐:从入门到精通系列学习路线书籍介绍 转自:http://blog.csdn.net/findsafety/article/details/52317506 很多时候我们都会不 ...
- php执行运算符
php执行运算符 简介 php 支持一个执行运算符:反引号(``).反引号(``)位于键盘Tab键左上方.php 将尝试将反引号中的内容作为外壳命令来执行,并将其输出信息返回(例如,可以赋给一个变量而 ...
- python spark 求解最大 最小 平均 中位数
rating_data_raw = sc.textFile("%s/ml-100k/u.data" % PATH) print rating_data_raw.first() nu ...
- LDA解决的问题
人类是怎么生成文档的呢?LDA的这三位作者在原始论文中给了一个简单的例子.比如假设事先给定了这几个主题:Arts.Budgets.Children.Education,然后通过学习训练,获取每个主题T ...
- java8新特性系列:[1]让你的eclipse支持java8
package com.anhui.jdk8; /** * 针对eclipse是否支持java8小测试 * MainClass * @author zhongzh * */ public class ...
- php 判断过去离现在几年的函数
function gettime($worktime){ $time=time(); $amount=date("Y",$time)-date("Y",strt ...
- 给统计人讲Python(1)_科学计算库-Numpy
本地代码是.ipynb格式的转换到博客上很麻烦,这里展示部分代码,了解更多可以查看我的git-hub:https://github.com/Yangami/Python-for-Statisticia ...
- Linux Shell Scripting Cookbook 读书笔记 6
wget,curl, tar, rsync wget ftp://example.com/somefile.img -t 5 -O download.img -o log -t表示重试的次数 -O指定 ...
- ROS-tutorials-launch-查看日志
前言:launch文件的作用是一次可以启动多个节点. 1.新建launch文件 在chapter2_tutorials包下新建launch文件夹,并在launch文件夹下新建chapter2.laun ...
- Aspose.Words进行Word替换(插入图片和水印)
由于最近一直在忙着做着Word打印模板的一些工作,就整理一些Asponse.Words对Word文档进行操作的资料. using System; using System.Collections.Ge ...