C#死锁的原理与排查方法详解

死锁的原理

线程死锁是指由于两个或者多个线程互相持有对方所需要的资源,并且互相等待对方释放资源,导致这些线程都处于等待状态,无法继续执行。如果线程都不主动释放所占有的资源,将产生死锁。如果死锁发生在UI线程,则会导致界面停止响应。.

死锁的条件

  • 互斥条件:线程对于所分配到的资源具有排它性,即一个资源只能被一个线程占用,直到被该线程释放
  • 请求和保持条件:一个线程因请求被占用资源而发生阻塞时,对已获得的资源保持不放。
  • 不剥夺条件:任何一个资源在没被该线程释放之前,任何其他线程都无法对他剥夺占用
  • 循环等待条件:当发生死锁时,所等待的线程必定会形成一个环路(类似于死循环),造成永久阻塞

死锁示例

C#死锁的原理与排查方法详解

如何避免死锁

  • 方法一. 破坏互斥条件
  • 方法二. 破坏环路等待条件
  • 方法三. 破坏不剥夺条件
  • 方法四. 破坏请求和保持条件

采用方法四举例:

C#死锁的原理与排查方法详解

排查方法

step1:在代码调试中,点击暂停按钮,然后vs就会自动定位到

C#死锁的原理与排查方法详解

step2:打开并行堆栈:然后点击指定方法名,进一步定位死锁语句

C#死锁的原理与排查方法详解

排查工具

可以采用第三方检测工具LockCop,检测死锁线程号,便于在非调试情况下查询死锁线程号,然后可以结合日志等信息排查死锁原因。

C#死锁的原理与排查方法详解