各位看官们。大家好,上一回中咱们说的是线程死锁的样例,这一回咱们继续说该样例。闲话休提,言归正转。让我们一起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语言实例--线程死锁三)的更多相关文章

  1. 一起talk C栗子吧(第一百一十二回:C语言实例--线程同步概述)

    各位看官们,大家好.上一回中咱们说的是线程间通信的样例,这一回咱们说的样例是:线程同步.闲话休提,言归正转.让我们一起talk C栗子吧! 看官们,提到同步.我想大家都不陌生,由于我们在前面章回中介绍 ...

  2. 一起talk C栗子吧(第十二回:C语言实例--单链表一)

    各位看官们,大家好.从今天開始,我们讲大型章回体科技小说 :C栗子.也就是C语言实例.闲话休提, 言归正转. 让我们一起talk C栗子吧! 看官们,上一回中咱们没有说详细的样例,并且是说了样例中的文 ...

  3. 第一百一十九节,JavaScript事件入门

    JavaScript事件入门 学习要点: 1.事件介绍 2.内联模型 3.脚本模型 4.事件处理函数 JavaScript事件是由访问Web页面的用户引起的一系列操作,例如:用户点击.当用户执行某些操 ...

  4. 一起talk C栗子吧(第二十二回:C语言实例--队列一)

    各位看官们,大家好,上一回中咱们说的是表达式求值的样例,该样例使用了栈,这一回咱们说的是栈的 兄弟:队列. 闲话休提,言归正转.让我们一起talk C栗子吧. 我们在这里说的队列是一种抽象的数据结构, ...

  5. 一起talk C栗子吧(第一百二十一回:C语言实例--线程知识体系图)

    各位看官们.大家好,上一回中咱们说的线程属性的样例.这一回咱们说的样例是:线程知识体系图.闲话休提.言归正转. 让我们一起talk C栗子吧! 我们在前面的章回中介绍了与线程相关的知识,在今天的章回中 ...

  6. “全栈2019”Java第一百一十二章:什么是闭包?

    难度 初级 学习时间 10分钟 适合人群 零基础 开发语言 Java 开发环境 JDK v11 IntelliJ IDEA v2018.3 文章原文链接 "全栈2019"Java第 ...

  7. 第三百一十九节,Django框架,文件上传

    第三百一十九节,Django框架,文件上传 1.自定义上传[推荐] 请求对象.FILES.get()获取上传文件的对象上传对象.name获取上传文件名称上传对象.chunks()获取上传数据包,字节码 ...

  8. [你必须知道的.NET]第十九回:对象创建始末(下)

    本文将介绍以下内容: 对象的创建过程 内存分配分析 内存布局研究 接上回[第十八回:对象创建始末(上)],继续对对象创建话题的讨论>>> 2.2 托管堆的内存分配机制 引用类型的实例 ...

  9. 第一百一十六节,JavaScript,DOM操作样式

    JavaScript,DOM操作样式 一.操作样式 CSS作为(X)HTML的辅助,可以增强页面的显示效果.但不是每个浏览器都能支持最新的CSS能力.CSS的能力和DOM级别密切相关,所以我们有必要检 ...

随机推荐

  1. Objective-C浮点数转化整数(向上取整、向下取整)

    Objective-C拓展了C,自然很多用法是和C一致的.比如浮点数转化成整数,就有以下四种情况. 1.简单粗暴,直接转化 float f = 1.5; int a; a = (int)f; NSLo ...

  2. Linux目录结构(二)

    Linux文件系统结的结构是树形结构,其入口从/开始,了解Linux文件系统的结构,对于我们需要掌握的基础知识点之一. 2.文件系统的组织结构简说: 当您使用Linux的时候,如果您通过ls -la ...

  3. java8-2-Lambda表达式

    java8的lambda表达式:使得代码更加紧凑:修改方法的能力:更好的支持多核处理(并行处理函数和filter\map\reduce) 例子1: java7中,list集合排序: public st ...

  4. LeetCode 42. Trapping Rain Water 【两种解法】(python排序遍历,C++ STL map存索引,时间复杂度O(nlogn))

    LeetCode 42. Trapping Rain Water Python解法 解题思路: 本思路需找到最高点左右遍历,时间复杂度O(nlogn),以下为向左遍历的过程. 将每一个点的高度和索引存 ...

  5. 1350 Taxi Cab Scheme DAG最小路径覆盖

    对于什么是DAG最小路径覆盖以及解题方法在我的另外的博客已经有了.http://www.cnblogs.com/Potato-lover/p/3980470.html 此题的题意: 公交车(出租车)车 ...

  6. 将自己的类封装为lib的方法

    前言: Windows API中所有的函数都包含在dll中,其中有3个最重要的DLL. (1)   Kernel32.dll 它包含那些用于管理内存.进程和线程的函数,例如CreateThread函数 ...

  7. iphone通讯录的备份与恢复

    最近在做iOS系统通讯录备份到服务器,并且可以从服务器中下载备份文件恢复到手机的功能,部分实现细节记录如下. 将iphone系统通讯录生成.vcf文件 ABAddressBookRef address ...

  8. WebLogic安装使用及常见问题

    WebLogic的下载与安装 下载地址: http://www.oracle.com/technetwork/middleware/weblogic/downloads/index.html fmw_ ...

  9. Swfit4.0中JSON与模型原生互转(JSONEncoder/JSONDecoder的使用)

    在Objective-C中,苹果并没有提供JSON转模型(模型转JSON)的接口,往往在开中需要添加第三库来处理JSON数据,比如:JsonModel.MJExtension.Mantle.JsonK ...

  10. Chernobyl’ Eagle on a Roof(鹰蛋坚固度)

    链接 Chernobyl’ Eagle on a Roof 题意 引用论文题意:有一堆共 M 个鹰蛋,一位教授想研究这些鹰蛋的坚硬度 E.他是通过不断从一幢 N 层的楼上向下扔鹰蛋来确定 E 的.当鹰 ...