C#中使用lock和Monitor控制多线程对资源的使用,最常见的生产者和消费者问题就是多线程同步和通信的经典例子。了解C#多线程的同步与通信。

一、关于lock和Monitor

lock可以把一段代码定义为互斥段(critical section),互斥段在一个时刻内只允许一个线程进入执行,而其它线程必须等待。格式定义如下:

lock(expression) statement_block

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

Monitor用于多线程公用一个对象时使线程共享资源的方案。Monitor必须和一个具体的对象相关联。

二、生产者和消费者问题

假设两个线程同时维护一个队列,如果一个线程对队列中更新元素,而另外一个线程从队列中获取元素,那么我们称更新元素的线程为生产者,称获取元素的线程为消费者。

1、被操作对象

   /// <summary>;
/// 被操作对象
/// </summary>;
public class Counter
{
//更新和读取的数字
private int numberOfCounter;
//读操作可执行标记,可以防止死锁的发生
private bool readFlag = false; public void Read()
{
//锁定后,其它读操作等待这一次读操作完成
lock (this)
{
//第一次之行为flase,进入等待
if (!readFlag)
{
try
{
//进入等待读,另一个线程写
Monitor.Wait(this);
}
catch (Exception ex)
{
Console.WriteLine(ex);
}
} Console.WriteLine("消费(获取): {0}", numberOfCounter); //重置,消费已经完成
readFlag = false;
Monitor.Pulse(this);
}
} public void Write(int number)
{
//锁定后,其它写操作等待这一次写操作完成
lock (this)
{
//第一次readFlag为flase,跳过执行下边的写
//如果当前正在读,等待读操作执行Monitor.Pulse
if (readFlag)
{
try
{
Monitor.Wait(this);
}
catch (Exception ex)
{
Console.WriteLine(ex);
}
}
numberOfCounter = number;
Console.WriteLine("生产(更新): {0}", numberOfCounter); //重置,生产已经完成
readFlag = true; //同步通过等待Pulse来完成
Monitor.Pulse(this);
}
}
}

2、生产者和消费者

   /// <summary>;
/// 生产者
/// </summary>
public class CounterWrite
{
Counter counter;
//生产者生产次数
int quantity = 1; public CounterWrite(Counter box, int request)
{
//构造函数
counter = box;
quantity = request;
} //生产者向操作对象更新信息
public void Write()
{
for (int i = 1; i &lt;= quantity; i++)
counter.Write(i);
}
} /// <summary>
/// 消费者
/// </summary>
public class CounterRead
{
Counter counter;
//生产者生产次数
int quantity = 1; public CounterRead(Counter box, int request)
{
//构造函数
counter = box;
quantity = request;
} //消费者从操作对象中获取信息
public void Read()
{
for (int i = 1; i &lt;= quantity; i++)
counter.Read();
}
}

3、线程操作

       Counter counter = new Counter();

            CounterRead read = new CounterRead(counter, 10);
CounterWrite write = new CounterWrite(counter, 10); Thread th1 = new Thread(new ThreadStart(read.Read));
Thread th2 = new Thread(new ThreadStart(write.Write)); th1.Start();
th2.Start(); th1.Join();
th2.Join(); Console.ReadLine();

通过lock锁定Counter对象的引用,初始readFlag为false控制线程1等待读取:Monitor.Wait(this),
线程2写入,然后更改readFlag,然后执行:Monitor.Pulse(this),通知等待队列中的线程请求对象状态已发生改变,
线程1锁定this,执行读操作,然后更改readFlag,线程1和线程2交互执行写读的操作。
同时因为readFlag的存在和交替更新,避免了死锁情况的发生。

C#多线程的同步与通信的更多相关文章

  1. Python 多线程、多进程 (二)之 多线程、同步、通信

    Python 多线程.多进程 (一)之 源码执行流程.GIL Python 多线程.多进程 (二)之 多线程.同步.通信 Python 多线程.多进程 (三)之 线程进程对比.多线程 一.python ...

  2. Qt 多线程同步与通信

    Qt 多线程同步与通信 1 多线程同步 Qt提供了以下几个类来完成这一点:QMutex.QMutexLocker.QSemphore.QWaitCondition. 当然可能还包含QReadWrite ...

  3. JAVASE02-Unit010: 多线程基础 、 TCP通信

    多线程基础 . TCP通信 * 当一个方法被synchronized修饰后,那么 * 该方法称为同步方法,即:多个线程不能同时 * 进入到方法内部执行. package day10; /** * 当多 ...

  4. 转载自~浮云比翼:Step by Step:Linux C多线程编程入门(基本API及多线程的同步与互斥)

    Step by Step:Linux C多线程编程入门(基本API及多线程的同步与互斥)   介绍:什么是线程,线程的优点是什么 线程在Unix系统下,通常被称为轻量级的进程,线程虽然不是进程,但却可 ...

  5. Step by Step:Linux C多线程编程入门(基本API及多线程的同步与互斥)

    介绍:什么是线程,线程的优点是什么 线程在Unix系统下,通常被称为轻量级的进程,线程虽然不是进程,但却可以看作是Unix进程的表亲,同一进程中的多条线程将共享该进程中的全部系统资源,如虚拟地址空间, ...

  6. Java多线程之线程的通信

    Java多线程之线程的通信 在总结多线程通信前先介绍一个概念:锁池.线程因为未拿到锁标记而发生的阻塞不同于前面五个基本状态中的阻塞,称为锁池.每个对象都有自己的锁池的空间,用于放置等待运行的线程.这些 ...

  7. java多线程与线程间通信

    转自(http://blog.csdn.net/jerrying0203/article/details/45563947) 本文学习并总结java多线程与线程间通信的原理和方法,内容涉及java线程 ...

  8. java第五节 多线程/多线程的同步

    /* 第五讲 多线程 了解进程和线程 在多任务系统中,每个独立执行的程序称为进程,也就是"正在进行的程序",我们现在使用的操作系统一般都是多任务的 即能够同时执行多个应用程序,实际 ...

  9. 扯扯python的多线程的同步锁 Lock RLock Semaphore Event Condition

    我想大家都知道python的gil限制,记得刚玩python那会,知道了有pypy和Cpython这样的解释器,当时听说是很猛,也就意味肯定是突破了gil的限制,最后经过多方面测试才知道,还是那德行… ...

随机推荐

  1. ajax+vue简单使用

    <script type="text/javascript" src="http://cdn.bootcss.com/vue/2.2.2/vue.min.js&qu ...

  2. IIS下配置Php+Mysql+zend的图文教程(完整版)

    网上有很多关于PHP在IIS下配置的教程,但都是一些很理性化的东西,我从里面整理出来这个教程 发出来为了方便参考,,有什么问题也可以大家一起交流,,如果有什么不对的地方,请指正.. 下面的教程都是在w ...

  3. 重复的DNA序列[哈希表] LeetCode.187

    所有 DNA 由一系列缩写为 A,C,G 和 T 的核苷酸组成,例如:"ACGAATTCCG".在研究 DNA 时,识别 DNA 中的重复序列有时会对研究非常有帮助. 编写一个函数 ...

  4. dapi 基于Django的轻量级测试平台六 怎样使用压测功能

    QQ群: GitHub:https://github.com/yjlch1016/dapi JMeter非GUI模式下: jmeter -n -t jmx脚本 -l jtl文件 -e -o 测试报告目 ...

  5. C#中的函数(三)参数传递及返回值

    接前面二篇,继续开始新的研究 前面忘了说什么是主调函数与被调函数 主调函数:执行调用其它函数语句所在的函数 被调函数:被其它函数所调用的函数 简单说就是一个是发起调用者,另一个是被调用者 写个小例子说 ...

  6. ActiveMQ消息可靠性-持久性

    三个方面保证消息的可靠性 1.消息的持久 2.事物 3.签收 一:PERSISTENT:持久性   参数说明:1.持久   2.非持久 Java里面设置持久化和非持久 持久: 将持久性设置为持久 宕机 ...

  7. Scrapy笔记09- 部署

    Scrapy笔记09- 部署 本篇主要介绍两种部署爬虫的方案.如果仅仅在开发调试的时候在本地部署跑起来是很容易的,不过要是生产环境,爬虫任务量大,并且持续时间长,那么还是建议使用专业的部署方法.主要是 ...

  8. django运行报错TypeError: object supporting the buffer API required

    运行django项目报错:TypeError: object supporting the buffer API required 解决方案: 将settings.py中数据库的密码改成字符串格式 源 ...

  9. ESP8266 LUA脚本语言开发: 准备工作-硬件说明

    原理图 实物图 硬件说明 开发板板载说明: 1.主控芯片: ESP8266_12F 1.温湿度传感器DHT11   与连接ESP8266        GPIO4引脚 2.一路继电器         ...

  10. Android 开发基础入门篇: 复制一个工程作为一个新的工程

    说明 咱们做项目很多时候都需要复制一份工程出来作为一个新的工程 把第一节的工程拷贝到这一节 修改工程名字 打开软件导入此工程 修改包名 第一节的时候说了,一个APP一个包名 自行添加修改 自行修改 自 ...