zoukankan      html  css  js  c++  java
  • C# lock,Monitor 介绍(多线程并发控制)

      这里介绍C# lock关键字它可以把一段代码定义为互斥段(critical section),互斥段在一个时刻内只允许一个线程进入执行,而其他线程必须等待。

    C#语言有很多值得学习的地方,这里我们主要介绍C# lock关键字,包括介绍Monitor 类锁定一个对象等方面。

    C#提供了一个lock关键字,它可以把一段代码定义为互斥段(critical section),互斥段在一个时刻内只允许一个线程进入执行,而其他线程必须等待。在C#中,C# lock关键字定义如下:

    lock(expression) statement_block

    expression代表你希望跟踪的对象,通常是对象引用。
    ◆如果你想保护一个类的实例,一般地,你可以使用this;
    ◆如果你想保护一个静态变量(如互斥代码段在一个静态方法内部),一般使用类名就可以了。

    而statement_block就是互斥段的代码,这段代码在一个时刻内只可能被一个线程执行。

    下面是一个使用C# lock关键字的典型例子,在注释里说明了C# lock关键字的用法和用途。

    示例如下:

    1. using System;
    2. using System.Threading;
    3. namespace ThreadSimple
    4. {
    5. internal class Account
    6. {
    7. int balance;
    8. Random r = new Random();
    9. internal Account(int initial)
    10. {
    11. balance = initial;
    12. }
    13. internal int Withdraw(int amount)
    14. {
    15. if (balance <0)
    16. {
    17. //如果balance小于0则抛出异常
    18. throw new Exception("Negative Balance");
    19. }
    20. //下面的代码保证在当前线程修改balance的值完成之前
    21. //不会有其他线程也执行这段代码来修改balance的值
    22. //因此,balance的值是不可能小于0 的
    23. lock (this)
    24. {
    25. Console.WriteLine("Current Thread:"+Thread.CurrentThread.Name);
    26. //如果没有lock关键字的保护,那么可能在执行完if的条件判断之后
    27. //另外一个线程却执行了balancebalance=balance-amount修改了balance的值
    28. //而这个修改对这个线程是不可见的,所以可能导致这时if的条件已经不成立了
    29. //但是,这个线程却继续执行balancebalance=balance-amount,所以导致balance可能小于0
    30. if (balance >= amount)
    31. {
    32. Thread.Sleep(5);
    33. balancebalance = balance - amount;
    34. return amount;
    35. }
    36. else
    37. {
    38. return 0; // transaction rejected
    39. }
    40. }
    41. }
    42. internal void DoTransactions()
    43. {
    44. for (int i = 0; i <100; i++)
    45. Withdraw(r.Next(-50, 100));
    46. }
    47. }
    48. internal class Test
    49. {
    50. static internal Thread[] threads = new Thread[10];
    51. public static void Main()
    52. {
    53. Account acc = new Account (0);
    54. for (int i = 0; i <10; i++)
    55. {
    56. Thread t = new Thread(new ThreadStart(acc.DoTransactions));
    57. threads[i] = t;
    58. }
    59. for (int i = 0; i <10; i++)
    60. threads[i].Name=i.ToString();
    61. for (int i = 0; i <10; i++)
    62. threads[i].Start();
    63. Console.ReadLine();
    64. }
    65. }
    66. }

    Monitor 类锁定一个对象

    当多线程公用一个对象时,也会出现和公用代码类似的问题,这种问题就不应该使用C# lock关键字了,这里需要用到System.Threading中的一个类Monitor,我们可以称之为监视器,Monitor提供了使线程共享资源的方案。

    Monitor类可以锁定一个对象,一个线程只有得到这把锁才可以对该对象进行操作。对象锁机制保证了在可能引起混乱的情况下一个 时刻只有一个线程可以访问这个对象。 Monitor必须和一个具体的对象相关联,但是由于它是一个静态的类,所以不能使用它来定义对象,而且它的所有方法都是静态的,不能使用对象来引用。下 面代码说明了使用Monitor锁定一个对象的情形:

    1. ......
    2. Queue oQueue=new Queue();
    3. ......
    4. Monitor.Enter(oQueue);
    5. ......//现在oQueue对象只能被当前线程操纵了
    6. Monitor.Exit(oQueue);//释放锁  

    如上所示,当一个线程调用Monitor.Enter()方法锁定一个对象时,这个对象就归它所有了,其它线程想要访问这个对象, 只有等待它使用 Monitor.Exit()方法释放锁。为了保证线程最终都能释放锁,你可以把Monitor.Exit()方法写在try-catch- finally结构中的finally代码块里。

  • 相关阅读:
    爱的感悟
    连点成图:享受创建图形的乐趣
    python实现动态更新远程机器列表的SSH登录脚本
    python生成数据库中所有表的DESC描述
    生活之美
    克服“测试怠惰”的习惯
    使用git和github托管个人项目
    连点成线
    一次合并数据库的经历
    python使用装饰器捕获异常
  • 原文地址:https://www.cnblogs.com/xyqCreator/p/2854441.html
Copyright © 2011-2022 走看看