在上一篇C#多线程之线程同步篇2中,我们主要学习了AutoResetEvent构造、ManualResetEventSlim构造和CountdownEvent构造,在这一篇中,我们将学习Barrier构造、ReaderWriterLockSlim构造和SpinWait构造。

七、使用Barrier构造

  在这一小节中,我们将学习一个比较有意思的同步构造:Barrier。Barrier构造可以帮助我们控制多个等待线程达到指定数量后,才发送通知信号,然后所有等待线程才能继续执行,并且在每次等待线程达到指定数量后,还能执行一个回调方法。具体步骤如下所示:

1、使用Visual Studio 2015创建一个新的控制台应用程序。

2、双击打开“Program.cs”文件,编写代码如下所示:

 using System;
using System.Threading;
using static System.Console;
using static System.Threading.Thread; namespace Recipe07
{
class Program
{
static Barrier barrier = new Barrier(, b => WriteLine($"End of phase {b.CurrentPhaseNumber + 1}")); static void PlayMusic(string name, string message, int seconds)
{
for(int i = ; i < ; i++)
{
WriteLine("----------------------------------------------");
Sleep(TimeSpan.FromSeconds(seconds));
WriteLine($"{name} starts to {message}");
Sleep(TimeSpan.FromSeconds(seconds));
WriteLine($"{name} finishes to {message}");
barrier.SignalAndWait();
} } static void Main(string[] args)
{
var t1 = new Thread(() => PlayMusic("the guitarist", "play an amazing solo", ));
var t2 = new Thread(() => PlayMusic("the singer", "sing his song", )); t1.Start();
t2.Start();
}
}
}

3、运行该控制台应用程序,运行效果如下图所示:

  在第10行代码处,我们创建了一个Barrier的实例barrier,并给其构造方法的“participantCount”参数赋值为2,表示barrier参与线程的数量为2,也就是说要有2个线程达到阻塞后,barrier才发送通知信号,其阻塞线程才能继续执行。第二个参数“postPhaseAction”是一个Action类型的委托,表示当阻塞线程达到规定数量后要执行的回调方法。

  在第28~29行代码处,我们创建了2个线程t1和t2,用于执行“PlayMusic”方法。t2线程首先执行到第21行代码处,在这一行代码中,我们在线程t2中调用了barrier的“SignalAndWait”方法,等待参与数量的线程达到构造方法指定的数量2时,才能继续执行,因为,在t2线程调用该方法时,只有一个线程t2被阻塞,没有达到规定数量2,所以,t2线程不能继续执行。当t1线程执行到第21行代码处时,也调用了barrier的“SignalAndWait”方法,这个时候等待线程的数量达到规定的数量2,所以t1和t2线程都能继续执行,并且在barrier的构造方法的第二个参数指定的回调方法也被执行。

  当两个线程执行“PlayMusic”方法的第二次循环时,过程与第一次一样,不在描述。

八、使用ReaderWriterLockSlim构造

  在这一小节中,我们将学习如何使用ReaderWriterLockSlim构造来线程安全地使用多线程读写集合中的数据。具体步骤如下所示:

1、使用Visual Studio 2015创建一个新的控制台应用程序。

2、双击打开“Program.cs”文件,编写代码如下所示:

 using System;
using System.Collections.Generic;
using System.Threading;
using static System.Console;
using static System.Threading.Thread; namespace Recipe08
{
class Program
{
// 表示用于管理资源访问的锁定状态,可实现多线程读取或进行独占式写入访问
static ReaderWriterLockSlim rw = new ReaderWriterLockSlim();
static Dictionary<int, int> items = new Dictionary<int, int>(); static void Read()
{
WriteLine("Reading contents of a dictionary");
while (true)
{
try
{
// 尝试进入读取模式锁定状态
rw.EnterReadLock();
foreach(var key in items.Keys)
{
Sleep(TimeSpan.FromSeconds(0.1));
}
}
finally
{
// 减少读取模式的递归计数,并在生成的计数为 0(零)时退出读取模式
rw.ExitReadLock();
}
}
} static void Write(string threadName)
{
while (true)
{
try
{
int newKey = new Random().Next();
// 尝试进入可升级模式锁定状态
rw.EnterUpgradeableReadLock();
if (!items.ContainsKey(newKey))
{
try
{
// 尝试进入写入模式锁定状态
rw.EnterWriteLock();
items[newKey] = ;
WriteLine($"New key {newKey} is added to a dictionary by a {threadName}");
}
finally
{
// 减少写入模式的递归计数,并在生成的计数为 0(零)时退出写入模式
rw.ExitWriteLock();
}
}
Sleep(TimeSpan.FromSeconds(0.1));
}
finally
{
// 减少可升级模式的递归计数,并在生成的计数为 0(零)时退出可升级模式
rw.ExitUpgradeableReadLock();
}
}
} static void Main(string[] args)
{
new Thread(Read) { IsBackground = true }.Start();
new Thread(Read) { IsBackground = true }.Start();
new Thread(Read) { IsBackground = true }.Start(); new Thread(() => Write("Thread 1")) { IsBackground = true }.Start();
new Thread(() => Write("Thread 2")) { IsBackground = true }.Start(); Sleep(TimeSpan.FromSeconds());
}
}
}

3、运行该控制台应用程序,运行效果(每次运行效果可能不同)如下图所示:

  在第73~75行代码处,我们创建了3个后台线程来读取集合中的数据。在第77~78行代码处,我们创建了2个后台线程向集合中写入数据。为了线程安全地对集合进行操作,我们使用为此场景专门设计的ReaderWriterLockSlim构造。该构造有两种类型的锁:读取模式锁和写入模式锁。读取模式锁允许多线程读取数据,写入模式锁阻塞其他线程的每一个操作直到写入模式锁被释放为止。

  有一个非常有趣的场景,当我们想获得一个读取模式锁从集合中读取一些数据,并根据这些数据获得一个写入模式锁以更新集合时,如果我们立即就获得锁定模式锁的话不仅消耗的时间多,而且还不允许我们读取数据,因为当我们获得一个写入模式锁的时候,集合就被锁定了。为了尽量减少这种时间的浪费,我们可以使用“EnterUpgradeableReadLock”方法获得读取模式锁来读取数据,如果读取完毕数据后,我们发现需要更新底层集合,那么我们可以使用“EnterWriteLock”升级我们的锁,然后快速执行写入操作并使用“ExitWriteLock”释放写入模式锁,最后使用“ExitUpgradeableReadLock”释放可升级模式锁。

  在上述代码中,我们获得一个随机数,然后获得一个读取模式锁,并检查该随机数是否已在集合中存在,如果不存在,我们升级该读取模式锁为写入模式锁,然后向集合中添加一个新的key。使用try/finally块是一个比较好的方式,它可以保证我们总能释放锁获得的锁。

九、使用SpinWait构造

  在这一小节中,我们将学习如何在不涉及kernel-mode构造的情况下等待一个线程的执行。另外还将介绍SpinWait构造,该构造是一种混合同步构造,主要用于设计在用户模式中等待一段时间后,然后将其切换到内核模式,以节省CUP时间。具体步骤如下所示:

1、使用Visual Studio 2015创建一个新的控制台应用程序。

2、双击打开“Program.cs”文件,编写代码如下所示:

 using System;
using System.Threading;
using static System.Console;
using static System.Threading.Thread; namespace Recipe09
{
class Program
{
static volatile bool isCompleted = false; static void UserModeWait()
{
while (!isCompleted)
{
Write(".");
}
WriteLine();
WriteLine("Waiting is complete");
} static void HybridSpinWait()
{
// 提供对基于自旋的等待的支持
var w = new SpinWait();
while (!isCompleted)
{
// 执行单一自旋
w.SpinOnce();
// 获取对 System.Threading.SpinWait.SpinOnce 的下一次调用是否将产生处理器,同时触发强制上下文切换
WriteLine(w.NextSpinWillYield);
}
WriteLine("Waiting is complete");
} static void Main(string[] args)
{
var t1 = new Thread(UserModeWait);
var t2 = new Thread(HybridSpinWait); WriteLine("Running user mode waiting");
t1.Start();
Sleep();
isCompleted = true;
Sleep(TimeSpan.FromSeconds());
isCompleted = false;
WriteLine("Running hybrid SpinWait construct waiting");
t2.Start();
Sleep();
isCompleted = true;
}
}
}

3、运行该控制台应用程序,运行效果(每次运行效果可能不同)如下图所示:

  在上述程序中,我们创建了一个线程执行一个无线循环20毫秒,直到在主线程中将isCompleted变量设置为true。我们可以将此时间设置为20-30秒,然后打开任务管理器,我们可以看到CPU的使用率比较高。

  我们使用volatile关键字声明了一个名为“isCompleted”的静态字段。volatile 关键字指示一个字段可以由多个同时执行的线程修改。声明为 volatile 的字段不受编译器优化(假定由单个线程访问)的限制。这样可以确保该字段在任何时间呈现的都是最新的值。

  然后,我们使用SpinWait版本,在第29行代码处,我们调用了SpinWait的“SpinOnce”方法,执行一次自旋。当SpinWait自旋达到一定次数后,如果有必要当前线程会让出底层的时间片并触发上下文切换。在这个版本中,如果我们将第49行代码的等待时间修改为20~30秒,然后打开任务管理器,可以发现CPU使用率是比较低的。

  至此,关于线程同步的知识就学习到这儿!

  源码下载

C#多线程之线程同步篇3的更多相关文章

  1. C#多线程之线程同步篇2

    在上一篇C#多线程之线程同步篇1中,我们主要学习了执行基本的原子操作.使用Mutex构造以及SemaphoreSlim构造,在这一篇中我们主要学习如何使用AutoResetEvent构造.Manual ...

  2. C#多线程之线程同步篇1

    在多线程(线程同步)中,我们将学习多线程中操作共享资源的技术,学习到的知识点如下所示: 执行基本的原子操作 使用Mutex构造 使用SemaphoreSlim构造 使用AutoResetEvent构造 ...

  3. C#多线程之线程池篇1

    在C#多线程之线程池篇中,我们将学习多线程访问共享资源的一些通用的技术,我们将学习到以下知识点: 在线程池中调用委托 在线程池中执行异步操作 线程池和并行度 实现取消选项 使用等待句柄和超时 使用计时 ...

  4. 关于Java多线程的线程同步和线程通信的一些小问题(顺便分享几篇高质量的博文)

    Java多线程的线程同步和线程通信的一些小问题(顺便分享几篇质量高的博文) 前言:在学习多线程时,遇到了一些问题,这里我将这些问题都分享出来,同时也分享了几篇其他博客主的博客,并且将我个人的理解也分享 ...

  5. C#多线程之线程池篇3

    在上一篇C#多线程之线程池篇2中,我们主要学习了线程池和并行度以及如何实现取消选项的相关知识.在这一篇中,我们主要学习如何使用等待句柄和超时.使用计时器和使用BackgroundWorker组件的相关 ...

  6. C#多线程之线程池篇2

    在上一篇C#多线程之线程池篇1中,我们主要学习了如何在线程池中调用委托以及如何在线程池中执行异步操作,在这篇中,我们将学习线程池和并行度.实现取消选项的相关知识. 三.线程池和并行度 在这一小节中,我 ...

  7. 重新想象 Windows 8 Store Apps (46) - 多线程之线程同步: Lock, Monitor, Interlocked, Mutex, ReaderWriterLock

    [源码下载] 重新想象 Windows 8 Store Apps (46) - 多线程之线程同步: Lock, Monitor, Interlocked, Mutex, ReaderWriterLoc ...

  8. 重新想象 Windows 8 Store Apps (47) - 多线程之线程同步: Semaphore, CountdownEvent, Barrier, ManualResetEvent, AutoResetEvent

    [源码下载] 重新想象 Windows 8 Store Apps (47) - 多线程之线程同步: Semaphore, CountdownEvent, Barrier, ManualResetEve ...

  9. IOS 多线程,线程同步的三种方式

    本文主要是讲述 IOS 多线程,线程同步的三种方式,更多IOS技术知识,请登陆疯狂软件教育官网. 一般情况下我们使用线程,在多个线程共同访问同一块资源.为保护线程资源的安全和线程访问的正确性. 在IO ...

随机推荐

  1. 利用snowfall.jquery.js实现爱心满屏飞

    小颖在上一篇一步一步教你用CSS画爱心中已经分享一种画爱心的方法,这次再分享一种方法用css画爱心,并利用snowfall.jquery.js实现爱心满屏飞的效果. 第一步: 利用伪元素before和 ...

  2. jsp中出现onclick函数提示Cannot return from outside a function or method

    在使用Myeclipse10部署完项目后,原先不出错的项目,会有红色的叉叉,JSP页面会提示onclick函数错误 Cannot return from outside a function or m ...

  3. Ubuntu 16.10 开启PHP错误提示

    两个步骤: 修改php.ini配置文件中的error_reporting 和 display_errors两地方内容: sudo vim /etc/php/7.0/apache2/php.ini er ...

  4. 史上最详细git教程

    题外话 虽然这个标题很惊悚,不过还是把你骗进来了,哈哈-各位看官不要着急,耐心往下看 Git是什么 Git是目前世界上最先进的分布式版本控制系统. SVN与Git的最主要的区别 SVN是集中式版本控制 ...

  5. ASP.NET SignaiR 实现消息的即时推送,并使用Push.js实现通知

    一.使用背景 1. SignalR是什么? ASP.NET SignalR 是为 ASP.NET 开发人员提供的一个库,可以简化开发人员将实时 Web 功能添加到应用程序的过程.实时 Web 功能是指 ...

  6. springmvc 多数据源 SSM java redis shiro ehcache 头像裁剪

    获取下载地址   QQ 313596790  A 调用摄像头拍照,自定义裁剪编辑头像 B 集成代码生成器 [正反双向](单表.主表.明细表.树形表,开发利器)+快速构建表单;  技术:31359679 ...

  7. 简单酷炫的canvas动画

    作为一个新人怀着激动而紧张的心情写了第一篇帖子还请大家多多支持,小弟在次拜谢. 驯鹿拉圣诞老人动画效果图如下 html如下: <div style="width:400px;heigh ...

  8. 敏捷测试模式之Scrum及其实践

    一.    敏捷开发模式简介 敏捷是近年来软件研发领域很火的一个词,采用敏捷开发模式的研发团队是越来越多了,尤其是敏捷模式中的Scrum更是佼佼者大行其道,这表明敏捷模式确有其好处,能给企业带来效率的 ...

  9. MongoDB学习笔记五—查询上

    数据准备 { , "goods_name" : "KD876", "createTime" : ISODate("2016-12- ...

  10. MySQL全文索引 FULLTEXT索引和like的区别

    1.概要 InnoDB引擎对FULLTEXT索引的支持是MySQL5.6新引入的特性,之前只有MyISAM引擎支持FULLTEXT索引.对于FULLTEXT索引的内容可以使用MATCH()-AGAIN ...