c# volatile 关键字的拾遗补漏

 更新时间:2020年10月28日 09:02:11   作者:精致码农 • 王亮  
这篇文章主要介绍了c# volatile 关键字的相关资料,帮助大家更好的理解和学习c#的相关知识,感兴趣的朋友可以了解下

要理解 C# 中的 volatile 关键字,就要先知道编译器背后的一个基本优化原理。比如对于下面这段代码:

public class Example
{
 public int x;
 public void DoWork()
 {
  x = 5;
  var y = x + 10;
  Debug.WriteLine("x = " +x + ", y = " +y);
 }
}

在 Release 模式下,编译器读取 x = 5 后紧接着读取 y = x + 10,在单线程思维模式下,编译器会认为 y 的值始终都是 15。所以编译器会把 y = x + 10 优化为 y = 15,避免每次读取 y 都执行一次 x + 5。但 x 字段的值可能在运行时被其它的线程修改,我们拿到的 y 值并不是通过最新修改的 x 计算得来的,y 的值永远都是 15

也就是说,编译器在 Release 模式下会对字段的访问进行优化,它假定字段都是由单个线程访问的,把与该字段相关的表达式运算结果编译成常量缓存起来,避免每次访问都重复运算。但这样就可能导致其它线程修改了字段值而当前线程却读取不到最新的字段值。为了防止编译器这么做,你就要让编译器用多线程思维去解读代码。告诉编译器字段的值可能会被其它线程修改,这种情况不要使用优化策略。而要做到这一点,就需要使用 volatile 关键字。

给类的字段添加 volatile 关键字,目的是告诉编译器该字段的值可能会被多个独立的线程改变,不要对该字段的访问进行优化。

使用 volatile 可以确保字段的值是可用的最新值,而且该值不会像非 volatile 字段值那样受到缓存的影响。好的做法是将每个可能被多个线程使用的字段标记为 volatile,以防止非预期的优化行为。

为了加深理解,我们来看一个实际的例子:

public class Worker
{
 private bool _shouldStop;

 public void DoWork()
 {
  bool work = false;
  // 注意:这里会被编译器优化为 while(true)
  while (!_shouldStop)
  {
   work = !work; // do sth.
  }
  Console.WriteLine("工作线程:正在终止...");
 }

 public void RequestStop()
 {
  _shouldStop = true;
 }
}

public class Program
{
 public static void Main()
 {
  var worker = new Worker();

  Console.WriteLine("主线程:启动工作线程...");
  var workerTask = Task.Run(worker.DoWork);

  // 等待 500 毫秒以确保工作线程已在执行
  Thread.Sleep(500);

  Console.WriteLine("主线程:请求终止工作线程...");
  worker.RequestStop();

  // 待待工作线程执行结束
  workerTask.Wait();
  //workerThread.Join();

  Console.WriteLine("主线程:工作线程已终止");
 }
}

在这个例子中,while (!_shouldStop) 会被编译器优化为 while(true)。我们可以看一下实际的运行效果来验证这一点。切换 Release 模式,按 Ctrl + F5 运行程序,运行效果始终如下:

程序运行后,虽然主线程在 500 毫秒后执行 RequestStop() 方法修改了 _shouldStop 的值,但工作线程始终都获取不到 _shouldStop 最新的值,也就永远都不会终止 while 循环。

我们修改一下程序,对 _shouldStop 字段加上 volatile 关键字:

public class Worker
{
 private volatile bool _shouldStop;

 public void DoWork()
 {
  bool work = false;
  // 获取的是最新的 _shouldStop 值
  while (!_shouldStop)
  {
   work = !work; // do sth.
  }
  Console.WriteLine("工作线程:正在终止...");
 }

 // ...(略)
}

此时在主线程调用 RequestStop() 方法后,工作线程便立即终止了,运行效果如下图所示:

这说明加了 volatile 关键字后,程序可以实时读取到字段的最新值。

注意,一定要切换为 Release 模式运行才能看到 volatile 发挥的作用,Debug 模式下即使添加了 volatile 关键字,编译器也是不会执行优化的。

当然,并不是所有的类型都可以使用 volatile 关键字修饰的,常见的使用 volatile 的类型是这些简单类型:sbyte, byte, short, ushort, int, uint, char, float 和 bool,其它的请查看参考链接。

作者:精致码农

出处:http://cnblogs.com/willick

联系:liam.wang@live.com

以上就是c# volatile 关键字的拾遗补漏的详细内容,更多关于c# volatile 关键字的资料请关注脚本之家其它相关文章!

相关文章

  • C# AE之返回上一级和下一级的实战操作

    C# AE之返回上一级和下一级的实战操作

    这篇文章主要介绍了C# AE之返回上一级和下一级的实战操作,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2021-01-01
  • C# Socket网络编程实例

    C# Socket网络编程实例

    这篇文章主要介绍了C# Socket网络编程实例,分析了Socket网络通信的原理与具体应用技巧,具有一定参考借鉴价值,需要的朋友可以参考下
    2015-01-01
  • C#设计模式之Builder生成器模式解决带老婆配置电脑问题实例

    C#设计模式之Builder生成器模式解决带老婆配置电脑问题实例

    这篇文章主要介绍了C#设计模式之Builder生成器模式解决带老婆配置电脑问题,简单介绍了生成器模式的概念、功能并结合具体实例形式分析了C#生成器模式解决配电脑问题的步骤与相关操作技巧,需要的朋友可以参考下
    2017-09-09
  • C#代码实现短信验证码接口示例

    C#代码实现短信验证码接口示例

    这篇文章主要为大家详细介绍了C#实现短信验证码接口示例代码,感兴趣的小伙伴们可以参考一下
    2016-08-08
  • C#使用GDI绘制直线的方法

    C#使用GDI绘制直线的方法

    这篇文章主要介绍了C#使用GDI绘制直线的方法,涉及C#中使用GDI绘图的相关技巧,需要的朋友可以参考下
    2015-04-04
  • C# ExecuteScalar()方法案例讲解

    C# ExecuteScalar()方法案例讲解

    这篇文章主要介绍了C# ExecuteScalar()方法案例讲解,本篇文章通过简要的案例,讲解了该项技术的了解与使用,以下就是详细内容,需要的朋友可以参考下
    2021-08-08
  • c#如何显式实现接口成员

    c#如何显式实现接口成员

    这篇文章主要介绍了c#如何显式实现接口成员,帮助大家更好的利用c#处理接口,感兴趣的朋友可以了解下
    2020-10-10
  • C#使用IronPython库调用Python脚本

    C#使用IronPython库调用Python脚本

    这篇文章介绍了C#使用IronPython库调用Python脚本的方法,文中通过示例代码介绍的非常详细。对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2022-06-06
  • C# ManagementObjectSearcher操作window案例详解

    C# ManagementObjectSearcher操作window案例详解

    这篇文章主要介绍了C# ManagementObjectSearcher操作window案例详解,本篇文章通过简要的案例,讲解了该项技术的了解与使用,以下就是详细内容,需要的朋友可以参考下
    2021-08-08
  • C# 多窗口委托通信的实现

    C# 多窗口委托通信的实现

    本文主要介绍了C# 多窗口委托通信的实现,窗口之间通信无非有两个方向,主窗口发送数据到副窗口,副窗口发送数据到主窗口,感兴趣的可以了解一下
    2022-03-03

最新评论