通过锁来实现同步

排它锁主要用来保证,在一段时间内,只有一个线程可以访问某一段代码。两种主要类型的排它锁是lock和Mutex。Lock和Mutex相比构造起来更方便,运行的也更快。但是Mutex可以在同一个机器上的不同进程使用。

Monitor.Enter和Monitor.Exit

C#中的lock关键字,实际上是Monitor.Enter,Monitor.Exist的一个简写。在.NET 1.0,2.0,3.0 版本的c#中,lock会被编译成如下代码:

        Monitor.Enter(_locker);try{if (_val2 != 0) Console.WriteLine(_val1 / _val2);_val2 = 0;}finally { Monitor.Exit(_locker); }

如果你没有调用Monitor.Enter而直接调用Monitor.Exit会引发异常。

LockTaken版本:

想象一下上面这段代码,如果再Monitor.Enter之后,try之前,线程出现了异常(比如被终止),在这种情况下,finally中的Exit方法就永远不会被执行,也就导致了这个锁不会被释放。为了避免这种情况,CLR 4.0的设计者重载了Monitor.Enter方法:

public static void Enter (object obj, ref bool lockTaken);

如果当前线程由于某些异常导致锁没有被获取到,lockTake值会为false,因此在CLR 4.0中,lock会被解释成如下代码:

        bool lockTaken = false;try{Monitor.Enter(_locker, ref lockTaken);// Do your stuff...}finally { if (lockTaken) Monitor.Exit(_locker); }

TryEnter

Monitor也提供了了一个TryEnter方法,允许你设置一个超时时间,避免当前线程长时间获取不到锁而一直等待。

选择正确的同步对象

你需要选择一个对所有线程都可见的对象进行lock(obj)来确保程序能够按照你的意图执行。如果比不了解C#语言中的某些特性,lock可能不会按照你 期望来执行。

  1. 由于字符串的驻留机制,lock("string")不是一个好的选择
  2. Lock一个值类型不是一个好的选择
  3. Lock(typeof(..))不是一个好的选择,因为System.Type的特性

什么时候使用lock

一个基本的规则,你需要对任意的写操作,或者可修改的字段进行lock。即使是一个赋值操作,或者累加操作,你也不能假设他是线程安全的。

例如下面代码不是线程安全的:

        class ThreadUnsafe{static int _x;static void Increment() { _x++; }static void Assign() { _x = 123; }}

你需要这样写:

        class ThreadSafe{static readonly object _locker = new object();static int _x;static void Increment() { lock (_locker) _x++; }static void Assign() { lock (_locker) _x = 123; }}

如果你看过一些BCL类库里面的实现,你可以能会发现,某些情况下会使用InterLocked类,而不是lock,我们会在后面介绍。

关于嵌套锁或者reentrant

你在阅读一些文档的时候,有的文档可能会说lock或者Monitor.Enter是reentrant(可重入的),那么我们如何理解reentrant呢?

想象下以下代码:

            lock (locker)lock (locker)lock (locker){// Do something...}

或者是:

            Monitor.Enter(locker); Monitor.Enter(locker); Monitor.Enter(locker);// Do something...Monitor.Exit(locker); Monitor.Exit(locker); Monitor.Exit(locker);

这种情况下,只有在最后一个exit执行后,或者执行了相应次数的Exit后,locker才是可获取的状态。

Mutex

Mutex像c#中的lock一样,但是在不同的进程中仍然可以使用。换句话说,Mutex是一个计算机级别的锁。因此获取这样一个锁要比Monitor慢很多。

示例代码:

using System;
using System.Threading.Tasks;
using System.Threading;namespace MultiThreadTest
{class OneAtATimePlease{static void Main(){// Naming a Mutex makes it available computer-wide. Use a name that's// unique to your company and application (e.g., include your URL).using (var mutex = new Mutex(false, "oreilly.com OneAtATimeDemo")){// Wait a few seconds if contended, in case another instance// of the program is still in the process of shutting down.if (!mutex.WaitOne(TimeSpan.FromSeconds(3), false)){Console.WriteLine("Another app instance is running. Bye!");return;}RunProgram();}}static void RunProgram(){Console.WriteLine("Running. Press Enter to exit");Console.ReadLine();}}
}

Semaphore

Monitor和Mutex都是排他锁,Semaphore我们常用的另外一种非排他的锁。

我们用它来实现这样一个例子:一个酒吧,最多能容纳3人,如果客满则需要等待,有客人离开,等待的人随时可以进来。

示例代码:

using System;
using System.Threading;class TheClub      // No door lists!
{static Semaphore _sem = new Semaphore(3, 3);    // Capacity of 3static void Main(){for (int i = 1; i <= 5; i++) new Thread(Enter).Start(i);Console.ReadLine();}static void Enter(object id){Console.WriteLine(id + " wants to enter");_sem.WaitOne();Console.WriteLine(id + " is in!");           // Only three threadsThread.Sleep(1000 * (int)id);               // can be here atConsole.WriteLine(id + " is leaving");       // a time._sem.Release();}
}

使用Semaphore需要调用者来控制访问资源,调用WaitOne来获取资源,通过Release来释放资源。开发者有责任确保资源能够正确释放。

Semaphore在限制同步访问的时候非常有用,它不会像Monitor或者Mutex那样当一个线程访问某些资源时,其它所有线程都需要等,而是设置一个缓冲区,允许最多多少个线程同时进行访问。

Semaphore也可以像Mutex一样,跨进程进行同步。

本节主要总结了使用锁进行同步,下一节将总结使用信号量进行同步。

转载于:https://www.cnblogs.com/myprogram/p/4924335.html

细说.NET中的多线程 (四 使用锁进行同步)相关推荐

  1. 【转】细说.NET中的多线程 (四 使用锁进行同步)

    通过锁来实现同步 排它锁主要用来保证,在一段时间内,只有一个线程可以访问某一段代码.两种主要类型的排它锁是lock和Mutex.Lock和Mutex相比构造起来更方便,运行的也更快.但是Mutex可以 ...

  2. 【转】细说.NET中的多线程 (五 使用信号量进行同步)

    上一节主要介绍了使用锁进行同步,本节主要介绍使用信号量进行同步 使用EventWaitHandle信号量进行同步 EventWaitHandle主要用于实现信号灯机制.信号灯主要用于通知等待的线程.主 ...

  3. 【转】细说.NET中的多线程 (二 线程池)

    上一章我们了解到,由于线程的创建,销毁都是需要耗费大量资源和时间的,开发者应该非常节约的使用线程资源.最好的办法是使用线程池,线程池能够避免当前进程中大量的线程导致操作系统不停的进行线程切换,当线程数 ...

  4. 【转】细说.NET 中的多线程 (一 概念)

    为什么使用多线程 1.使用户界面能够及时响应用户的输入 当某个应用程序在进行大量运算时候,为了保证应用程序能够随时响应客户的输入,这个时候我们往往需要让大量运算和响应用户输入这两个行为在不同的线程中进 ...

  5. Java多线程(四)线程锁

    6.锁 由于多个线程是共同占有所属进程的资源和地址空间的,那么就会存在一个问题: 如果多个线程要同时访问某个资源,怎么处理? 在Java并发编程中,经常遇到多个线程访问同一个 共享资源 ,这时候作为开 ...

  6. java 锁竞争_Java多线程中的竞争条件、锁以及同步的概念

    竞争条件 1.竞争条件: 在java多线程中,当两个或以上的线程对同一个数据进行操作的时候,可能会产生"竞争条件"的现象.这种现象产生的根本原因是因为多个线程在对同一个数据进行操作 ...

  7. 【转】细说.NET中的多线程 (六 使用MemoryBarrier,Volatile进行同步)

    上一节介绍了使用信号量进行同步,本节主要介绍一些非阻塞同步的方法.本节主要介绍MemoryBarrier,volatile,Interlocked. MemoryBarriers 本文简单的介绍一下这 ...

  8. 【转】细说.NET中的多线程 (三 使用Task)

    上一节我们介绍了线程池相关的概念以及用法.我们可以发现ThreadPool. QueueUserWorkItem是一种起了线程之后就不管了的做法.但是实际应用过程,我们往往会有更多的需求,比如如何更简 ...

  9. (四)Java中的多线程之间实现同步+多线程并发同步

    一.什么是线程安全问题 为什么有线程安全问题? 当多个线程同时共享同一个全局变量或静态变量,做写的操作(修改变量值)时,可能会发生数据冲突问题,也就是线程安全问题.但是做读操作时不会发生数据冲突问题. ...

最新文章

  1. 【APP接口开发】chrome浏览器DHC工具安装使用(亲测有效)
  2. 【Codevs1322】单词矩阵
  3. 2020 我的C++的学习之路
  4. SBT模版(Size Balanced Tree)
  5. springCloud - 第7篇 - 配置文件管理中心 ( SpringCloud Config )
  6. java定义一个矩阵的类_java写入一个矩阵,如何编程求该矩阵的秩
  7. 最方便的字符设备驱动的写法
  8. eclipse导入后将普通项目变为java项目build path,no action avilia
  9. 一个简单漂亮的Java博客系统
  10. python缠论代码_缠论dll(czsc - 缠中说禅技术分析工具)
  11. 数学建模(十)博弈论
  12. matlab 有源高滤波器,基于MATLAB有源滤波器的研究设计.ppt
  13. 为什么“电路中容抗和阻抗相等时,也就是谐振时,电路呈阻性?”谢谢
  14. PC-DIMS测量中遇到了不可解决的问题
  15. 互联网时代如何做好网络营销
  16. 基于破壳漏洞的蠕虫实践
  17. matlab magnify程序,magnifymatlab源程序
  18. Fliqlo——翻页时钟屏保(最新版本,附有链接)
  19. Android M 新控件了解学习
  20. 计算机毕业设计Java课程资源管理平台(源码+mysql数据库+系统+lw文档)

热门文章

  1. 反卷积,上采样,上池化的理解
  2. 详解机器学习的凸优化、图神经网络、强化学习、贝叶斯方法等四大主题
  3. SGD、Adam优化器
  4. YOLO3 动漫人脸识别
  5. RHCSA学习笔记-基础操作命令
  6. HTML5 Canvas中实现文字链接
  7. Android-----使用Button特效selector+shape
  8. wps中将文档输出为pdf_‎
  9. 如何将项目发布到阿里云_尚硅谷基于阿里云搭建数据仓库(实时)项目视频发布...
  10. 懒加载和预加载的区别_类的动态创建(ro,rw)amp; 懒加载类和非懒加载类底层加载的区别 amp; 类和分类的搭配分析...