通过局部变量访问volatile字段

And*_*niy 6 java concurrency performance volatile

这个问题在某种程度上有所延续和扩展,因为我认为完美的问题:如何分配给局部变量有帮助?

这个问题基础上Item 71Effective Java,它建议在目的引入局部变量来提高性能volatile字段访问:

private volatile FieldType field;  
FieldType getField() {  
    FieldType result = field;  
    if (result == null) { // First check (no locking)  
        synchronized(this) {   
        result = field;  
        if (result == null) // Second check (with locking)  
            field = result = computeFieldValue();  
        }  
    }  
    return result;  
}
Run Code Online (Sandbox Code Playgroud)

所以,我的问题更常见:

我们应该始终volatile通过将其值分配给局部变量来访问字段吗? (为了存档最佳性能).

即一些成语:

  1. 我们有一些volatile领域,称之为volatileField;

  2. 如果我们想在多线程方法中读取它的值,我们应该:

    1. 创建相同类型的局部变量: localVolatileVariable
    2. 指定volatile字段的值: localVolatileVariable = volatileField
    3. 从本地副本读取值,例如:

      if (localVolatileVariable != null) { ... }
      
      Run Code Online (Sandbox Code Playgroud)

jta*_*orn 4

如果您计划执行任何类型的多步骤逻辑(当然假设该字段是可变的),则必须将 易失性变量分配给本地字段。

例如:

volatile String _field;

public int getFieldLength() {
  String tmp = _field;
  if(tmp != null) {
    return tmp.length();
  }
  return 0;
}
Run Code Online (Sandbox Code Playgroud)

如果您没有使用 的本地副本_field,则该值可能会在“if”测试和“length()”方法调用之间发生变化,从而可能导致 NPE。

除了通过不执行一次以上的易失性读取来提高速度的明显好处之外。