一起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级别密切相关,所以我们有必要检 ...
随机推荐
- Jenkins project
1.project name 这个作为git clone的target folder 2.Multiple SCMs 建立2个git类型的操作,相互独立. additional Behaviors 设 ...
- hdoj--3339--In Action(最短路+01背包)
In Action Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others) Total ...
- 利用python开发的flappy bird 游戏
python 中 pygame模块能让我们很方便的编写游戏,16年我用python 仿制了flappy bird 游戏,下面是游戏的完整代码以及素材,分享给大家. 第一个python文件,flappy ...
- 5个对话框和FileStream:文件流
1.private void button1_Click(object sender, EventArgs e) { colorDialog1.ShowDialog();//显示颜色选择器 panel ...
- VS2015启动显示ID为XXXX的进程当前未运行
解决办法:在启动项目根目录下用文本编辑器打开Web项目下的{X}.csproj文件,然后查找 <WebProjectProperties>,将这一对标签之间的内容全部删除,然后再打开项目就 ...
- lua单链表实现
List = {} --创建一个节点 function List.new(val) return {pnext = nil, value = val} end --往一个节点后添加一个节点 funct ...
- [Offer收割]编程练习赛35
有歧义的号码 #include<stdio.h> #include<string.h> #include<stdlib.h> int cmp(const void ...
- hihoCoder挑战赛31
#1595 : Numbers 时间限制:8000ms 单点时限:1000ms 内存限制:256MB 描述 给定n个整数常数c[1], c[2], ..., c[n]和一个整数k.现在需要给2k个整数 ...
- P1410 子序列
题目描述 给定一个长度为N(N为偶数)的序列,问能否将其划分为两个长度为N/2的严格递增子序列, 输入输出格式 输入格式: 若干行,每行表示一组数据.对于每组数据,首先输入一个整数N,表示序列的长度. ...
- 解决Android单个dex文件不能超过65535个方法问题
一.找坑:谷歌规定单个dex文件中的方法不能超过65536的限制 我们编写项目过程中在工程的lib文件夹下引用的第三方插件jar包太多或者项目过大,编译运行时就有可能报出com.android.dex ...