死锁产生的4个必要条件:
    1、互斥:一个资源同一时刻只允许一个线程进行访问。
    2、占有未释放:一个线程占有资源,且没有释放资源。
    3、不可抢占:一个已经占有资源的线程无法抢占到其他线程拥有的资源。
    4、循环等待:两个或者两个以上的线程,本身拥有资源,不释放资源,并且同时尝试获得其他线程所持有的资源,这种资源的申请关系形成一个闭环的链条。

死锁的避免:

关于死锁的避免,仁者见仁智者见智。

主要还是从造成死锁的四个条件入手,四个条件不能满足,就不会死锁。

以下是我本人的一些方法:

1.线程等待时(wait)给予一个默认的等待时间

2.线程之间资源避免相互申请对方的资源,可以通过一些容器来控制并发,比如blockqueue,等等一些线程安全的容器

3.尽量避免线程在等待的同时申请资源

4.死锁检测,一个线程在等待一段时间后还没有获得资源就放弃申请。对等待时间进行检测。

一个死锁的例子:

  1. # 线程死锁,在classA和classB相互调用对方的方法,并且相互等待对方释放资源
  2.  
  3. class ClassA:
  4. def __init__(self):
  5. self.lock=threading.RLock()
  6.  
  7. # 得到classA的锁,试图得到classB的锁
  8. def infoA(self, b):
  9. self.lock.acquire()
  10. time.sleep(10)
  11. b.info()
  12. self.lock.release()
  13.  
  14. def info(self):
  15. self.lock.acquire()
  16. print("this is ClassA info")
  17. self.lock.release()
  18.  
  19. class ClassB:
  20. def __init__(self):
  21. self.lock = threading.RLock()
  22.  
  23. # 得到classB的锁,试图得到classA的锁
  24. def infoB(self, a):
  25. self.lock.acquire()
  26. time.sleep(10)
  27. a.info()
  28. self.lock.release()
  29.  
  30. def info(self):
  31. self.lock.acquire()
  32. print("this is ClassB info")
  33. self.lock.release()
  34.  
  35. ca=ClassA()
  36. cb=ClassB()
  37.  
  38. def funA():
  39. ca.infoA(cb)
  40. def funB():
  41. cb.infoB(ca)
  42.  
  43. # 函数调用不是线程,不会死锁
  44. # funA()
  45. # funB()
  46.  
  47. t1=threading.Thread(target=funA).start()
  48. t2=threading.Thread(target=funB).start()

说明:

线程 状态      
classA 获得classA的锁,sleep(classA的锁未释放) 申请classB的锁 死锁  
classB 获得classB的锁,sleep(classB的锁未释放) 申请classA的锁 死锁  
         

Python线程-死锁的更多相关文章

  1. python线程死锁与递归锁

    死锁现象 所谓死锁: 是指两个或两个以上的进程或线程在执行过程中,因争夺资源而造成的一种互相等待的现象,若无外力作用,它们都将无法推进下去. 此时称系统处于死锁状态或系统产生了死锁,这些永远在互相等待 ...

  2. Python并发编程-进程 线程 同步锁 线程死锁和递归锁

    进程是最小的资源单位,线程是最小的执行单位 一.进程 进程:就是一个程序在一个数据集上的一次动态执行过程. 进程由三部分组成: 1.程序:我们编写的程序用来描述进程要完成哪些功能以及如何完成 2.数据 ...

  3. Python多线程,线程死锁及解决,生产者与消费者问题

    1.Thread类 普通调用 t = Thread(target=test, args=(i,)) # test为目标函数名, 若函数需要参数将其以元组形 # 式赋给args, 若无参数可不写 t.s ...

  4. python——线程与多线程进阶

    之前我们已经学会如何在代码块中创建新的线程去执行我们要同步执行的多个任务,但是线程的世界远不止如此.接下来,我们要介绍的是整个threading模块.threading基于Java的线程模型设计.锁( ...

  5. python线程同步原语--源码阅读

    前面两篇文章,写了python线程同步原语的基本应用.下面这篇文章主要是通过阅读源码来了解这几个类的内部原理和是怎么协同一起工作来实现python多线程的. 相关文章链接:python同步原语--线程 ...

  6. python 线程、多线程

    复习进程知识: python:主进程,至少有一个主线程 启动一个新的子进程:Process,pool 给每一个进程设定一下执行的任务:传一个函数+函数的参数 如果是进程池:map函数:传入一个任务函数 ...

  7. python线程入门

    目录 python线程入门 线程与进程 线程 总结 参考 python线程入门 正常情况下,我们在启动一个程序的时候.这个程序会先启动一个进程,启动之后这个进程会启动起来一个线程.这个线程再去处理事务 ...

  8. [ python ] 线程的操作

    目录 (见右侧目录栏导航) - 1. 前言    - 1.1 进程    - 1.2 有了进程为什么要有线程    - 1.3 线程的出现    - 1.4 进程和线程的关系    - 1.5 线程的 ...

  9. Python3 进程 线程 同步锁 线程死锁和递归锁

    进程是最小的资源单位,线程是最小的执行单位 一.进程 进程:就是一个程序在一个数据集上的一次动态执行过程. 进程由三部分组成: 1.程序:我们编写的程序用来描述进程要完成哪些功能以及如何完成 2.数据 ...

随机推荐

  1. P1058 车厢重组

    题目描述 在一个旧式的火车站旁边有一座桥,其桥面可以绕河中心的桥墩水平旋转.一个车站的职工发现桥的长度最多能容纳两节车厢,如果将桥旋转 \(180\) 度,则可以把相邻两节车厢的位置交换,用这种方法可 ...

  2. 基于AutoIt3的打印机安装

    #Region ;**** 编译指令由 AutoIt3Wrapper 选项编译窗口创建 **** #AutoIt3Wrapper_Icon=favicon.ico #AutoIt3Wrapper_Co ...

  3. springboot配置大全

    此配置大全是在官方开发者文档中看到的,地址:https://docs.spring.io/spring-boot/docs/1.5.6.RELEASE/reference/html/common-ap ...

  4. H3C查看系统启动配置文件

  5. webpack4.0基本配置,超简单!

    最近复习了一下webpack,使用的是4.0版本. 下图是基本目录结构,最后留有代码地址,有兴趣可以去看看. 直接上代码(依赖未完全使用): 项目的所有依赖都可以安装,每个都有详细的注释.] cons ...

  6. codeforces 1165F1/F2 二分好题

    Codeforces 1165F1/F2 二分好题 传送门:https://codeforces.com/contest/1165/problem/F2 题意: 有n种物品,你对于第i个物品,你需要买 ...

  7. codeforces 1185G1 状压dp

    codeforces 1185G1. Playlist for Polycarp (easy version)(动态规划) 传送门:https://codeforces.com/contest/118 ...

  8. slim中的请求URI

    请求 URI 每个 HTTP 请求都有一个识别被请求的应用程序资源的 URI .HTTP 请求 URI 分为这几部分: Scheme (e.g. http or https) Host (e.g. e ...

  9. mybatis精讲(六)--二级缓存

    目录 简介 配置 源码 CachingExecutor 自定义二级缓存 # 加入战队 微信公众号 简介 上一章节我们简单了解了二级缓存的配置.今天我们详细分析下二级缓存以及为什么不建议使用二级缓存. ...

  10. [译文] 为什么你在 C# 里总是应该使用 "var" 关键字

    [译文] Why You Should Always Use the 'var' Keyword in C# (为什么你总是应该在 C# 里使用 "var" 关键字) Using ...