这章的内容是真的有意思

特别是代码。

先贴上代码:

class Program
{
static void Main(string[] args)
{
object lock1 = new object(); object lock2 = new object(); new Thread(() => LockTooMuch(lock1, lock2)).Start(); lock(lock2)
{
Thread.Sleep(); Console.WriteLine("Monitor.TryEnter allow not to get stuck,returning false after a specified timer is elapsed"); if(Monitor.TryEnter(lock1,5))
{
Console.WriteLine("Acquired a protected resource succesfully"); }
else
{
Console.WriteLine("Timerout cquiring a resource"); }
} new Thread(() => LockTooMuch(lock1, lock2)).Start(); lock (lock2)
{
Console.WriteLine("This will be deadlock");
Thread.Sleep();
lock (lock1)
{
Console.WriteLine("Acquired a protected resource succesfully");
} }
Console.ReadKey();
}
static void LockTooMuch(object Lock1,object Lock2)
{
lock(Lock1)
{
Thread.Sleep(1); lock (Lock2)
{ } }
}
}

说实话,这章代码,我一开始没看明白。真的。

有些时候真的就是难者不会会者不难。

咱慢慢来来说。

咱先从这一部分开始:

   static  void LockTooMuch(object Lock1,object  Lock2)
{
lock(Lock1)
{
Thread.Sleep(); lock (Lock2)
{ } }
}

这一部分的代码很好理解:

传入两个引用类型的object参数。

然后是代码主要部分先是

锁死lock1

然后再lock1里面锁死lock2.

这个部分的锁死的代码,也就意味着解锁的顺序,lock1等待lock2执行完毕才会解锁lock1.

本章说避免锁死。那么什么是锁死?

那么我们来原书的代码:

    new Thread(() => LockTooMuch(lock1, lock2)).Start();
lock (lock2)
{
Console.WriteLine("This will be deadlock");
Thread.Sleep();
lock (lock1)
{
Console.WriteLine("Acquired a protected resource succesfully");
}
}

这一部分的代码加上静态的LockTooMatch的方法运行就会锁死。

我们按照这个思路清理一下代码:

 static void Main(string[] args)
{
object lock1 = new object(); object lock2 = new object(); new Thread(() => LockTooMuch(lock1, lock2)).Start(); lock (lock2)
{
Console.WriteLine("This will be deadlock");
Thread.Sleep();
lock (lock1)
{
Console.WriteLine("Acquired a protected resource succesfully");
} }
Console.ReadKey();
}
static void LockTooMuch(object Lock1,object Lock2)
{
lock(Lock1)
{
Thread.Sleep(); lock (Lock2)
{ } }
}

see

永远锁死

为什么只会执行到这里呢?

  Console.WriteLine("This will be deadlock");

一步一步的

首先程序新开线程运行LockTooMuch,然后再主线程锁LOCK2输出内容,线程等待1S之后再锁LOCK1 在输出语句。

             lock (lock2)
{
Console.WriteLine("This will be deadlock");
Thread.Sleep();
lock (lock1)
{
Console.WriteLine("Acquired a protected resource succesfully");
} }

按照我们之前说的东西,L2必须等待L1结束才会释放。

这是主线程的部分。

下面是方法的部分

 static  void LockTooMuch(object Lock1,object  Lock2)
{
lock(Lock1)
{
Thread.Sleep(); lock (Lock2)
{ } }
}

同样也是L1等待L2的结束才会是否。

主线程和新开线程是同时的。

也就是会有竞争。

那么执行的顺序是什么呢?

很简单

主线程新开线程

主线锁LOCK2

新开线程执行锁LOCK1

主线程输出

新开线程等待1s

主线程等待1s

新开线程锁LOCK2但是失败

主线程锁LOCK1但是失败

程序停止继续向下运行

为什么主线程,新线程锁LOCK为什么会失败呢?

新开线程锁lock2会失败是因为主线程在用lock2而且还没有用完。

主线程锁lock1会失败是应为新开线程做用lock1还么有用完。

所以会锁死。

ok 我们知道锁死的原因,下面我们通过Monitor.TryEnter来避免锁死。

Monitor.TryEnter是作用什么呢?

TryEnter有三个重载方法。

常用的是带有等待时间的。

作用就是在规定的时间去尝试解锁资源,或者获取资源。如果获取成功就返回true,不成功就返回false。这个代码总会返回结果。超时也会返回不过是false,也就说锁死了也会有办法来处理,不会尴尬的停在那里了

我们看看代码:

 new Thread(() => LockTooMuch(lock1, lock2)).Start();

            lock(lock2)
{
Thread.Sleep(); Console.WriteLine("Monitor.TryEnter allow not to get stuck,returning false after a specified timer is elapsed"); if(Monitor.TryEnter(lock1,5))
{
Console.WriteLine("Acquired a protected resource succesfully"); }
else
{
Console.WriteLine("Timerout cquiring a resource"); }
}

ok 清理下会锁死的代码,来运行代码

执行了else内的代码,说明尝试解锁lock1失败了,和上面那个会死锁的代码一样的结果,但是这个却可以继续持续其他代码,不会卡在死锁的部分。

我们来看看执行的过程【大家可以在VS中 使用断点,逐语句等方式来观察线程的运行顺序】

主线程新开线程

主线锁LOCK2

新开线程执行锁LOCK1

主线程等待1s

新开线程等待1s

主线程输出语句

新开线程锁LOCK2但是失败

主线程尝试解锁LOCK1,5s内。(成功返回true 失败返回fasle 超时返回false)但是失败

主线程输出else内容

程序结束

执行的顺序和死锁的顺序差不多,但是结果却不一样的因为Monitor.TryEnter避免的死锁的尴尬地步,程序多线程中出现死锁,使用monitor.tryenter会很好的避免死锁。

《C#多线程编程实战》1.11 Monitor.TryEnter()避免死锁的更多相关文章

  1. C#多线程编程实战(二)

    1.1 简介 为了防止一个应用程序控制CPU而导致其他应用程序和操作系统本身永远被挂起这一可能情况,操作系统不得不使用某种方式将物理计算分割为一些虚拟的进程,并给予每个执行程序一定量的计算能力.此外操 ...

  2. Java多线程编程实战指南(核心篇)读书笔记(五)

    (尊重劳动成果,转载请注明出处:http://blog.csdn.net/qq_25827845/article/details/76730459冷血之心的博客) 博主准备恶补一番Java高并发编程相 ...

  3. ASP.Net教程系列:多线程编程实战(一)

    Web开发中使用多线程可以增强用户体验,尤其是多用户.多任务.海量数据和资源紧张的情况下.所以我们的ASP.Net教程设立多线程编程实战专题.下面这些代码范例都是入门级的,希望对对大家学习ASP.Ne ...

  4. Java多线程编程实战读书笔记(一)

    多线程的基础概念本人在学习多线程的时候发现一本书——java多线程编程实战指南.整理了一下书中的概念制作成了思维导图的形式.按照书中的章节整理,并添加一些个人的理解.

  5. Java多线程编程实战指南(核心篇)读书笔记(四)

    (尊重劳动成果,转载请注明出处:http://blog.csdn.net/qq_25827845/article/details/76690961冷血之心的博客) 博主准备恶补一番Java高并发编程相 ...

  6. Java多线程编程实战指南(核心篇)读书笔记(三)

    (尊重劳动成果,转载请注明出处:http://blog.csdn.net/qq_25827845/article/details/76686044冷血之心的博客) 博主准备恶补一番Java高并发编程相 ...

  7. Java多线程编程实战指南(核心篇)读书笔记(二)

    (尊重劳动成果,转载请注明出处:http://blog.csdn.net/qq_25827845/article/details/76651408冷血之心的博客) 博主准备恶补一番Java高并发编程相 ...

  8. Java多线程编程实战指南(核心篇)读书笔记(一)

    (尊重劳动成果,转载请注明出处:http://blog.csdn.net/qq_25827845/article/details/76422930冷血之心的博客) 博主准备恶补一番Java高并发编程相 ...

  9. 《Java多线程编程实战指南(核心篇)》阅读笔记

    <Java多线程编程实战指南(核心篇)>阅读笔记 */--> <Java多线程编程实战指南(核心篇)>阅读笔记 Table of Contents 1. 线程概念 1.1 ...

随机推荐

  1. 1039 Course List for Student

    题意:给出K门课程(编号1~K)以及报名该课程的学生,然后有N个学生查询,对于每一个查询,输出该学生所报的相关课程编号,且要求编号按增序输出. 思路:题目不难,解析略.(本来用map直接映射,用STL ...

  2. Python Twisted系列教程16:Twisted 进程守护

    作者:dave@http://krondo.com/twisted-daemonologie/  译者: Cheng Luo 你可以从”第一部分 Twist理论基础“开始阅读:也可以从”Twisted ...

  3. day5心得

    import 模块 1.定义: 模块:用来从逻辑上组织python代码(变量.函数.类.逻辑:实现一个功能),本质就是.py结尾的python文件(文件名test.py 模块名:test) 2导入方法 ...

  4. 利用JS 在网页上获取并显示当前日期 星期

    下边的HTML代码,可以取出日期与星期 <html><body><h1><script language=JavaScript>var d, s = & ...

  5. leetcode524

    public class Solution { public string FindLongestWord(string s, IList<string> d) { string long ...

  6. ClientDataSet1.LoadFromStream

    lStream3.Position := 0;        lDataSet.LoadFromStream(lStream3);

  7. Super Jumping! Jumping! Jumping(最大递增子序列的和)

    Nowadays, a kind of chess game called “Super Jumping! Jumping! Jumping!” is very popular in HDU. May ...

  8. 【转】3D图形引擎(DX9): FX

    做图形程序已经很多年了,积累了一些经验.来这个论坛也有好几年了,遇到问题的时候,也会在这里发帖子问,也很感谢大家的回答与帮助,希望能多认识一些朋友,大家多交流.分享.做这款项目也有蛮多年了,终于上线了 ...

  9. 从SQL Server中清除msdb备份和恢复记录

    正如我在前面的技巧“您的数据库上次恢复是什么时候呢?”中提到的,SQL Server使msdb数据库内系统表中的备份和恢复记录保持激活状态.没有正常的维护,这些系统表将变得很大,从而导致对于msdb数 ...

  10. google/dense_hash_map

    这个库使用时需要注意的地方: 1.在插入数据之前,需要先调用set_empty_key()设置一个空Key,Key的值可以为任意符合类型的.但请注意之后插入的Key不能和空Key相同,否则会abort ...