NewBorn

凡心所向,素履以往;生如逆旅,一苇以航

导航

C# 特性之volatile

C# 2.0特性:Volatile

volatile 关键字指示一个字段可以由多个同时执行的线程修改。 出于性能原因,编译器,运行时系统甚至硬件都可能重新排列对存储器位置的读取和写入。 声明了 volatile 的字段不进行这些优化。 添加 volatile 修饰符可确保所有线程观察易失性写入操作(由任何其他线程执行)时的观察顺序与写入操作的执行顺序一致。 不确保从所有执行线程整体来看时所有易失性写入操作均按执行顺序排序。

volatile 关键字可应用于以下类型的字段:

  • 引用类型。
  • 指针类型(在不安全的上下文中)。 请注意,虽然指针本身可以是可变的,但是它指向的对象不能是可变的。 换句话说,不能声明“指向可变对象的指针”。
  • 简单类型,如 sbytebyteshortushortintuintcharfloat 和 bool
  • 具有以下基本类型之一的 enum 类型:bytesbyteshortushortint 或 uint
  • 已知为引用类型的泛型类型参数。
  • IntPtr 和 UIntPtr

其他类型(包括 double 和 long)无法标记为 volatile,因为对这些类型的字段的读取和写入不能保证是原子的。 若要保护对这些类型字段的多线程访问,请使用 Interlocked 类成员或使用 lock 语句保护访问权限。

volatile 关键字只能应用于 class 或 struct 的字段。 不能将局部变量声明为 volatile

 

下面的示例说明如何将公共字段变量声明为 volatile

class VolatileTest
{
    public volatile int sharedStorage;

    public void Test(int _i)
    {
        sharedStorage = _i;
    }
}

下面的示例演示如何创建辅助线程,并用它与主线程并行执行处理。

 1 public class Worker
 2 {
 3     // This method is called when the thread is started.
 4     public void DoWork()
 5     {
 6         bool work = false;
 7         while (!_shouldStop)
 8         {
 9             work = !work; // simulate some work
10         }
11         Console.WriteLine("Worker thread: terminating gracefully.");
12     }
13     public void RequestStop()
14     {
15         _shouldStop = true;
16     }
17     // Keyword volatile is used as a hint to the compiler that this data
18     // member is accessed by multiple threads.
19     private volatile bool _shouldStop;
20 }
21 
22 public class WorkerThreadExample
23 {
24     public static void Main()
25     {
26         // Create the worker thread object. This does not start the thread.
27         Worker workerObject = new Worker();
28         Thread workerThread = new Thread(workerObject.DoWork);
29 
30         // Start the worker thread.
31         workerThread.Start();
32         Console.WriteLine("Main thread: starting worker thread...");
33 
34         // Loop until the worker thread activates.
35         while (!workerThread.IsAlive)
36             ;
37 
38         // Put the main thread to sleep for 500 milliseconds to
39         // allow the worker thread to do some work.
40         Thread.Sleep(500);
41 
42         // Request that the worker thread stop itself.
43         workerObject.RequestStop();
44 
45         // Use the Thread.Join method to block the current thread 
46         // until the object's thread terminates.
47         workerThread.Join();
48         Console.WriteLine("Main thread: worker thread has terminated.");
49     }
50     // Sample output:
51     // Main thread: starting worker thread...
52     // Worker thread: terminating gracefully.
53     // Main thread: worker thread has terminated.
54 }

将 volatile 修饰符添加到 _shouldStop 的声明后,将始终获得相同的结果(类似于前面代码中显示的片段)。 但是,如果 _shouldStop 成员上没有该修饰符,则行为是不可预测的。 DoWork 方法可能会优化成员访问,从而导致读取陈旧数据。 鉴于多线程编程的性质,读取陈旧数据的次数是不可预测的。 不同的程序运行会产生一些不同的结果。

posted on 2019-06-06 13:09  K丶Formular  阅读(409)  评论(0编辑  收藏  举报