2019-03-20 11:37:56 +08:00
|
|
|
|
# 83. 明智审慎的使用延迟初始化
|
|
|
|
|
|
|
|
|
|
延迟初始化是延迟字段的初始化,直到需要它的值。如果不需要该值,则不会初始化字段。这种技术既适用于静态字段,也适用于实例字段。虽然延迟初始化主要是一种优化,但是它也可以用于破坏类中的有害循环和实例初始化 [Bloch05, Puzzle 51]。
|
|
|
|
|
|
2019-03-20 11:39:38 +08:00
|
|
|
|
与大多数优化一样,延迟初始化的最佳建议是「除非需要,否则不要这样做」(详见第 67 条)。延迟初始化是一把双刃剑。它降低了初始化类或创建实例的成本,代价是增加了访问延迟初始化字段的成本。根据这些字段中最终需要初始化的部分、初始化它们的开销以及初始化后访问每个字段的频率,延迟初始化实际上会损害性能(就像许多「优化」一样)。
|
2019-03-20 11:37:56 +08:00
|
|
|
|
|
|
|
|
|
延迟初始化也有它的用途。如果一个字段只在类的一小部分实例上访问,并且初始化该字段的代价很高,那么延迟初始化可能是值得的。唯一确定的方法是以使用和不使用延迟初始化的效果对比来度量类的性能。
|
|
|
|
|
|
|
|
|
|
在存在多个线程的情况下,使用延迟初始化很棘手。如果两个或多个线程共享一个延迟初始化的字段,那么必须使用某种形式的同步,否则会导致严重的错误(详见第 78 条)。本条目讨论的所有初始化技术都是线程安全的。
|
|
|
|
|
|
|
|
|
|
**在大多数情况下,常规初始化优于延迟初始化。** 下面是一个使用常规初始化的实例字段的典型声明。注意 final 修饰符的使用(详见第 17 条):
|
|
|
|
|
|
|
|
|
|
**如果您使用延迟初始化来取代初始化的循环(circularity),请使用同步访问器**,因为它是最简单、最清晰的替代方法:
|
|
|
|
|
|
|
|
|
|
```java
|
|
|
|
|
// Lazy initialization of instance field - synchronized accessor
|
|
|
|
|
private FieldType field;
|
|
|
|
|
private synchronized FieldType getField() {
|
|
|
|
|
if (field == null)
|
|
|
|
|
field = computeFieldValue();
|
|
|
|
|
return field;
|
|
|
|
|
}
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
这两种习惯用法(使用同步访问器进行常规初始化和延迟初始化)在应用于静态字段时都没有改变,只是在字段和访问器声明中添加了 static 修饰符。
|
|
|
|
|
|
|
|
|
|
**如果需要在静态字段上使用延迟初始化来提高性能,使用 lazy initialization holder class 模式。** 这个用法可保证一个类在使用之前不会被初始化 [JLS, 12.4.1]。它是这样的:
|
|
|
|
|
|
|
|
|
|
```java
|
|
|
|
|
// Lazy initialization holder class idiom for static fields
|
|
|
|
|
private static class FieldHolder {
|
|
|
|
|
static final FieldType field = computeFieldValue();
|
|
|
|
|
}
|
|
|
|
|
private static FieldType getField() { return FieldHolder.field; }
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
第一次调用 getField 时,它执行 FieldHolder.field,导致初始化 FieldHolder 类。这个习惯用法的优点是 getField 方法不是同步的,只执行字段访问,所以延迟初始化实际上不会增加访问成本。典型的 VM 只会同步字段访问来初始化类。初始化类之后,VM 会对代码进行修补,这样对字段的后续访问就不会涉及任何测试或同步。
|
|
|
|
|
|
|
|
|
|
如果需要使用延迟初始化来提高实例字段的性能,请使用双重检查模式。这个模式避免了初始化后访问字段时的锁定成本(详见第 79 条)。这个模式背后的思想是两次检查字段的值(因此得名 double check):一次没有锁定,然后,如果字段没有初始化,第二次使用锁定。只有当第二次检查指示字段未初始化时,调用才初始化字段。由于初始化字段后没有锁定,因此将字段声明为 volatile 非常重要(详见第 78 条)。下面是这个模式的示例:
|
|
|
|
|
|
|
|
|
|
```java
|
|
|
|
|
// Double-check idiom for lazy initialization of instance fields
|
|
|
|
|
private volatile FieldType field;
|
|
|
|
|
private FieldType getField() {
|
|
|
|
|
FieldType result = field;
|
|
|
|
|
if (result == null) { // First check (no locking)
|
|
|
|
|
synchronized(this) {
|
|
|
|
|
if (field == null) // Second check (with locking)
|
|
|
|
|
field = result = computeFieldValue();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
这段代码可能看起来有点复杂。特别是不清楚是否需要局部变量(result)。该变量的作用是确保 field 在已经初始化的情况下只读取一次。
|
|
|
|
|
|
|
|
|
|
虽然不是严格必需的,但这可能会提高性能,而且与低级并发编程相比,这更优雅。在我的机器上,上述方法的速度大约是没有局部变量版本的 1.4 倍。虽然您也可以将双重检查模式应用于静态字段,但是没有理由这样做:the lazy initialization holder class idiom 是更好的选择。
|
|
|
|
|
|
|
|
|
|
双重检查模式的两个变体值得注意。有时候,您可能需要延迟初始化一个实例字段,该字段可以容忍重复初始化。如果您发现自己处于这种情况,您可以使用双重检查模式的变体来避免第二个检查。毫无疑问,这就是所谓的「单检查」模式。它是这样的。注意,field 仍然声明为 volatile:
|
|
|
|
|
|
|
|
|
|
```java
|
|
|
|
|
// Single-check idiom - can cause repeated initialization!
|
|
|
|
|
private volatile FieldType field;
|
|
|
|
|
private FieldType getField() {
|
|
|
|
|
FieldType result = field;
|
|
|
|
|
if (result == null)
|
|
|
|
|
field = result = computeFieldValue();
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
本条目中讨论的所有初始化技术都适用于基本字段和对象引用字段。当双检查或单检查模式应用于数值基本类型字段时,将根据 0(数值基本类型变量的默认值)而不是 null 检查字段的值。
|
|
|
|
|
|
|
|
|
|
如果您不关心每个线程是否都会重新计算字段的值,并且字段的类型是 long 或 double 之外的基本类型,那么您可以选择在单检查模式中从字段声明中删除 volatile 修饰符。这种变体称为原生单检查模式。它加快了某些架构上的字段访问速度,代价是需要额外的初始化(每个访问该字段的线程最多需要一个初始化)。这绝对是一种奇特的技术,不是日常使用的。
|
|
|
|
|
|
|
|
|
|
总之,您应该正常初始化大多数字段,而不是延迟初始化。如果必须延迟初始化字段以实现性能目标或为了破坏友有害的初始化循环,则使用适当的延迟初始化技术。对于字段,使用双重检查模式;对于静态字段,则应该使用the lazy initialization holder class idiom。例如,可以容忍重复初始化的实例字段,您还可以考虑单检查模式。
|