使用Mutex类-互斥锁

owned为true,互斥锁的初始状态就是被主线程所获取,否则处于未获取状态

name为定义的互斥锁名称,在整个操作系统只有一个命名未CSharpThreadingCookbook的互斥量,一个线程得到,其他线程就无法得到这个互斥量了,只能等待。

const string mutexName = "CSharpThreadingCookbook";
public Mutex(bool initiallyOwned, string mutexName);
public Mutex();

WaitOne():获取,mut.ReleaseMutex():释放

public override bool WaitOne()
public static Mutex mutex = new Mutex();

//获取互斥锁,已经被其他线程获取,挂起等待。
mut.WaitOne();
//代码
//...
//... //释放锁
mut.ReleaseMutex();

SemaphoreSlim类,限制同一资源被同时访问的线程数量

有10个线程同时去访问数据库,借助信号类限制访问数据的并发线程数量,当有4个线程同时正在访问资源,其他线程必须等待,直至有资源执行完被释放。

class Program
{
static SemaphoreSlim _seamp = new SemaphoreSlim(4);
static void Main(string[] args)
{
for (int i = ; i <=; i++)
{
int j = i;
int sencondsToWait = + * i;
var t = new Thread(()=> { method(j.ToString(),sencondsToWait); });
t.Start();
}
Console.ReadLine();
}
static void method(string name,int senconds)
{
Console.WriteLine($"{name} waits to access a database");
_seamp.Wait();
Console.WriteLine($"{name} is completed");
Thread.Sleep(TimeSpan.FromSeconds(senconds));
_seamp.Release();
}
}

ManualResetEvent:线程间相互通信,一个线程开始一个活动,结束后告诉另一个线程可以进行执行了。

通过构造函数来控制 ManualResetEvent 的初始状态。

1.当初始值未false时,为非终止状态;_mre.WaitOne()阻塞线程,不允许线程访问下边的语句

static ManualResetEvent _mre = new ManualResetEvent(false);

2.当初始值未true时,为终止状态;_mre.WaitOne()允许线程访问下边的语句

static ManualResetEvent _mre = new ManualResetEvent(true);

在非终止状态时怎么让线程继续执行,怎么再让它停下来,这就要用了set()和Reset()方法了

  • 把非终止状态改为终止状态用Set()方法,告诉线程活动可以继续执行了(打开大门)
  • 把终止状态改为非终止状态用Reset()方法,阻塞线程直到再调用 Set()方法(关闭大门直至打开)

场景:

一个线程进行一个活动后,调用WaitOne()阻塞线程,等待其他线程给我信号我再继续执行。

另一个线程执行Set(),告诉正在等我执行完的线程,你可以执行了。随后Reset(),没有来得及执行的线程只能继续等待,直至再次接收信号。

CountDownEvent:信号类,等待直到一定数量的操作完成。

创建一个CountDownEvent实例,在构造函数中指定了当两个操作完成时会发出信号

static CountDownEvent _countDown = new CountDownEvent();
static void method()
{
Console.WriteLine("");
_countDown.Signal();
} var t1=new thread(()=>method());
t1.Start();
_countDown.Wait();
_countDown.dispose();
如果_countDown.Signal()没有达到指定的次数,那么_countDown.Wait()会一直等待,所以上面代码将会一值等待。

Barrier:像一个屏障,把所有线程任务的阶段隔离开来,当前阶段所有线程不完成,不会开始下一个阶段。

当屏障接收到所有的参与者,participantCount: 参与线程的数量,完成某个阶段的信号后,就会执行“阶段后代码”,

然后barrier.CurrentPhaseNumber会递增其值,表示要运行新的阶段,然后再解除阻塞每一个参与者的Task实例,然后所有参与者开始下一个阶段。

让【多个任务】在【不同阶段】同步一次(碰面),在继续往后执行。

        //同步两个线程,以及指定回调函数
static Barrier _barrier = new Barrier(, b => Console.WriteLine($"{b.CurrentPhaseNumber+1}当前阶段线程都执行完了,碰面一次") );
static void Main(string[] args)
{ var t1 = new Thread(() => Method_1());
var t2 = new Thread(() => Method_2());
t1.Start();
t2.Start();
Console.ReadLine();
} /// <summary>
/// 线程1要执行的方法
/// </summary>
public static void Method_1()
{
//模拟阶段一
Thread.Sleep(TimeSpan.FromSeconds());
Console.WriteLine("【阶段一】,线程【1】执行完毕");
_barrier.SignalAndWait(); //模拟阶段二
Thread.Sleep(TimeSpan.FromSeconds());
Console.WriteLine("【阶段二】,线程【1】执行完毕");
_barrier.SignalAndWait();
} /// <summary>
/// 线程2要执行的方法
/// </summary>
public static void Method_2()
{
//模拟阶段一
Thread.Sleep(TimeSpan.FromSeconds());
Console.WriteLine("【阶段一】,线程【2】执行完毕");
_barrier.SignalAndWait(); //模拟阶段二
Thread.Sleep(TimeSpan.FromSeconds());
Console.WriteLine("【阶段二】,线程【2】执行完毕");
_barrier.SignalAndWait(); }

运行结果:

【C#多线程编程实战笔记】二、 线程同步的更多相关文章

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

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

  2. C#多线程编程实战(二):线程同步

    2.1 简介 竞争条件:多个线程同时使用共享对象.需要同步这些线程使得共享对象的操作能够以正确的顺序执行 线程同步问题:多线程的执行并没有正确的同步,当一个线程执行递增和递减操作时,其他线程需要依次等 ...

  3. 【C#多线程编程实战笔记】一、 线程基础

    创建线程 Thread :所执行的方法不能有参数. class Program { static void Main(string[] args) { Console.WriteLine(" ...

  4. C#多线程编程实战1.6线程优先级

    using System;using System.Collections.Generic;using System.Linq;using System.Text;using System.Threa ...

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

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

  6. Python GUI之tkinter窗口视窗教程大集合(看这篇就够了) JAVA日志的前世今生 .NET MVC采用SignalR更新在线用户数 C#多线程编程系列(五)- 使用任务并行库 C#多线程编程系列(三)- 线程同步 C#多线程编程系列(二)- 线程基础 C#多线程编程系列(一)- 简介

    Python GUI之tkinter窗口视窗教程大集合(看这篇就够了) 一.前言 由于本篇文章较长,所以下面给出内容目录方便跳转阅读,当然也可以用博客页面最右侧的文章目录导航栏进行跳转查阅. 一.前言 ...

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

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

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

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

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

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

随机推荐

  1. Python零基础学习系列之一--初识计算机!

    1-1.计算机概念: Computer: 原指专门负责计算的人,后来演变成特指计算设备,译为"计算机" 计算机的概念: 计算机是能够根据一组指令操作数据的机器. A compute ...

  2. create pfile from spfile;

    sql>create pfile from spfile; 生成的文件在$ORACLE_HOME/dbs/下边    和spfile在同一个目录下 但是名字已经变成了init$oracle_si ...

  3. 在 overlay 中运行容器 - 每天5分钟玩转 Docker 容器技术(51)

    上一节我们创建了 overlay 网络 ov_net1,今天将运行一个 busybox 容器并连接到 ov_net1: 查看容器的网络配置: bbox1 有两个网络接口 eth0 和 eth1.eth ...

  4. Github 开源:使用升讯威 Mapper( Sheng.Mapper)与 AutoMapper 互补,大幅提高开发效率!

    Github 地址:https://github.com/iccb1013/Sheng.Mapper 在上一篇幅中,简单介绍了 升讯威 Mapper( Sheng.Mapper)[http://www ...

  5. hdu--4148--Length of S(n)

    #include<iostream> #include<string> #include<cstring> void priTable(); using names ...

  6. /etc/shadow,/etc/passwd,/etc/shadow,/etc/passwd文件的内容解释

    1.1 /etc/passwd文件内容格式           该目录存储的是操作系统用户信息,该文件为所有用户可见 用户名: 密码 : uid  : gid :用户描述:主目录:登陆shell 举个 ...

  7. MQ选型对比文档

    几种MQ产品说明:     ZeroMQ :  扩展性好,开发比较灵活,采用C语言实现,实际上他只是一个socket库的重新封装,如果我们做为消息队列使用,需要开发大量的代码    RabbitMQ  ...

  8. ArrayList 和Vector ,HashTable和HashMap异同

    相同点: 1.都实现了List接口(List接口继承自Collection接口) 2.有序集合,数据可重复,可按索引号取值(而HashSet无序,不可重复) 不同点: 1.Vector是线程安全的,而 ...

  9. 专治不会看源码的毛病--spring源码解析AOP篇

    昨天有个大牛说我啰嗦,眼光比较细碎,看不到重点.太他爷爷的有道理了!要说看人品,还是女孩子强一些.原来记得看到一个男孩子的抱怨,说怎么两人刚刚开始在一起,女孩子在心里就已经和他过完了一辈子.哥哥们,不 ...

  10. noip2016普及组 题解

    T1 大水题,不解释 上考场代码 #include <algorithm> #include <cstdio> using namespace std; int main() ...