各位看官们。大家好,上一回中咱们说的是线程死锁的样例,这一回咱们继续说该样例。闲话休提,言归正转。让我们一起talk C栗子吧!

看官们,由于篇幅的原因我们在上一回仅仅介绍了死锁发生的第一种原因,今天我们将介绍死锁发生的另外一种原因,而且该原因中的伪代码转换为实际的C语言代码

为了方便,我们使用前面章回中演示相互排斥量的代码,在该代码的基础上做一些小改动来演示死锁。代码例如以下:

首先定义两个相互排斥量,相互排斥量是全局变量。方便线程使用。

  1. #if MUTEX_ENABLE
  2. pthread_mutex_t mutex_value1;
  3. pthread_mutex_t mutex_value2;
  4. #endif

接下来在主进程中(也就是main函数)初始化两个相互排斥量:

  1. res = pthread_mutex_init(&mutex_value1,NULL);
  2. res = pthread_mutex_init(&mutex_value2,NULL);

在主进程的最后还要记得释放与相互排斥量相关的资源:

  1. #if MUTEX_ENABLE
  2. //destroy mutex
  3. res = pthread_mutex_destroy(&mutex_value1);
  4. res = pthread_mutex_destroy(&mutex_value2);
  5. #endif

我们分别改动两个线程的执行函数,该段代码是核心代码,请大家细致阅读:

  1. // the first thread function
  2. void *thread_func1(void *param)
  3. {
  4. int i = 0;
  5. int res = 0;
  6. pthread_t thread_id;
  7. thread_id = pthread_self();
  8. printf("Thread ID::%u -----------S---------- \n",(unsigned int)thread_id);
  9. while(i++ < 4)
  10. {
  11. #if MUTEX_ENABLE
  12. res = pthread_mutex_lock(&mutex_value1); // mutex1 is locking
  13. if(res != 0)
  14. {
  15. printf(" mutex1 lock failed \n");
  16. }
  17. #endif
  18. read_data("Thread_1");
  19. #if MUTEX_ENABLE
  20. res = pthread_mutex_lock(&mutex_value2); //mutex2 is locking
  21. if(res != 0)
  22. {
  23. printf(" mutex2 lock failed \n");
  24. }
  25. #endif
  26. #if MUTEX_ENABLE
  27. res = pthread_mutex_unlock(&mutex_value2);
  28. if(res != 0)
  29. {
  30. printf(" mutex2 unlock failed \n");
  31. }
  32. res = pthread_mutex_unlock(&mutex_value1);
  33. if(res != 0)
  34. {
  35. printf(" mutex1 unlock failed \n");
  36. }
  37. #endif
  38. sleep(2);
  39. }
  40. printf("Thread ID::%u -----------E---------- \n",(unsigned int)thread_id);
  41. pthread_exit(&status); // end the thread
  42. }
  1. // the second thread function
  2. void *thread_func2(void *param)
  3. {
  4. int i = 0;
  5. int res = 0;
  6. pthread_t thread_id;
  7. thread_id = pthread_self();
  8. printf("Thread ID::%u -----------S---------- \n",(unsigned int)thread_id);
  9. while(i++ < 4)
  10. {
  11. #if MUTEX_ENABLE
  12. res = pthread_mutex_lock(&mutex_value2); //mutex 2 is locking
  13. if(res != 0)
  14. {
  15. printf(" mutex2 lock failed \n");
  16. }
  17. #endif
  18. write_data("Thread_2");
  19. #if MUTEX_ENABLE
  20. res = pthread_mutex_lock(&mutex_value1); //mutex 1 is locking
  21. if(res != 0)
  22. {
  23. printf(" mutex1 lock failed \n");
  24. }
  25. #endif
  26. #if MUTEX_ENABLE
  27. res = pthread_mutex_unlock(&mutex_value1);
  28. if(res != 0)
  29. {
  30. printf(" mutex1 unlock failed \n");
  31. }
  32. #endif
  33. #if MUTEX_ENABLE
  34. res = pthread_mutex_unlock(&mutex_value2);
  35. if(res != 0)
  36. {
  37. printf(" mutex2 unlock failed \n");
  38. }
  39. #endif
  40. sleep(1);
  41. }
  42. printf("Thread ID::%u -----------E---------- \n",(unsigned int)thread_id);
  43. pthread_exit(&status); // end the thread
  44. }

我们执行上面的程序,能够得到下面结果:

  1. Create first thread //创建第一个线程
  2. Create second thread //创建第二个线程
  3. Thread ID::3076344640 -----------S----------
  4. [Thread_1] start reading data //第一个线程读取数据,同一时候对相互排斥量一加锁
  5. Thread ID::3067951936 -----------S----------
  6. [Thread_2] start writing data //第二个线程改动数据,同一时候对相互排斥量二加锁
  7. [Thread_1] data = 0
  8. [Thread_1] end reading data //第一个线程读取数据结束,同一时候等待相互排斥量二被解锁
  9. [Thread_2] data = 1
  10. [Thread_2] end writing data //第二个线程改动数据结束。同一时候等待相互排斥量一被解锁
  11. 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. Intellij IDEA社区版打包Maven项目成war包,并部署到tomcat上

    转自:https://blog.csdn.net/yums467/article/details/51660683 需求分析 我们利用 Intellij idea社区版IDE开发了一个maven的sp ...

  2. Eigen3

    Eigen用源码的方式提供给用户使用,在使用时只需要包含Eigen的头文件即可进行使用. Eigen: C++开源矩阵计算工具——Eigen的简单用法 http://blog.csdn.net/aug ...

  3. Python生成器实现杨辉三角打印

    def triangles(): c = [1] while 1: yield c a,b=[0]+c,c+[0] c=[a[i]+b[i] for i in range(len(a))] n = 0 ...

  4. vue2.0.js

     数据的渲染.数据同步  组件化.模块化 路由    ajax  数据流 Vue.js学习资源        中文官网:http://cn.vuejs.org/             源码:http ...

  5. hdu1829 A Bug's Life 基础种类并查集

    题目的大意可以理解为:A爱B,B爱C ……给出一系列爱恋的关系,推断有没有同性恋. 思路是把相同性别的归为一个集合(等价类),异性的异性为同性. #include<iostream> #i ...

  6. matlab中的@函数

    原文链接:http://blog.sina.com.cn/s/blog_5e73a8fc0100t9yg.html 这是个函数句柄 @(x,y) 表示未知数是x和y punct     - Funct ...

  7. 347. 前K个高频元素

    题目描述 给定一个非空的整数数组,返回其中出现频率前 k 高的元素. 示例 1: 输入: nums = [1,1,1,2,2,3], k = 2 输出: [1,2] 示例 2: 输入: nums = ...

  8. vc++图像保存,重绘

    新建mfc应用程序,单文档 增加绘图 分别增加命令响应 添加成员变量UINIT 图形可以运行,如何保存呢?(一个集合类,CPtArt) 用一个类的对象来保存一个图形的三个要素 所以插入一个新的类(通常 ...

  9. springboot版本控制

    HandlerMapping通过继承InitializingBean接口在完成实例后,扫描所有的Controller和标识RequestMapping的方法,缓存这个映射对应关系.然后在应用运行的时候 ...

  10. 路飞学城Python-Day153

    Scrapy核心组件