zoukankan      html  css  js  c++  java
  • Effective Java 之-----关于延迟初始化

    1、大多数情况下,正常的初始化要优先于延迟初始化。

    private final FieldType field = computeFieldValue();
    

    2、如果利用延迟优化来破坏初始化的循环,就要使用同步访问方法,因为它是最简单、最清楚的替代方法。

    private FieldType field;
    synchronized FieldType getField(){
        if(field == null){
            field = computeFieldValue();
        }
        return field;
    }
    

    3、如果出于性能考虑需要对静态域使用延迟初始化,就使用lazy initialization holder class 模式,保证类要到被用到的时候才会被初始化。

    private static class FieldHolder{
        static final FieldType field = computeFieldValue();
    }
    static FieldType getField(){
        return FieldHolder.field;
    }
    

      当getField被第一次调用时,第一次读取FieldHolder.field,导致FieldHolder得到初始化。

    4、如果出于性能考虑需要对实例域使用延迟初始化,就使用双重检查模式(double check idiom),这种模式避免了在域被初始化之后访问这个域时的开销。

    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;
    }
    

      两次检查域值,第一次检查时没有锁定,看这个域是否被初始化了;第二次检查时锁定,只有当第二次检查时表明这个域没有被初始化,才会调用computeFieldValue方法对这个域进行初始化。因为如果域已经被初始化就不会有锁定,域被声明为volatile很重要。

    5、如果可以接收重复初始化,也可以考虑使用单重检查模式(single check idiom)

    private volatile FieldType field;
    private FieldType getField(){
        FieldType result = field;
            if(result == null){
                field = result = computeFieldValue();
        return result;
    }

     

  • 相关阅读:
    Codeforces 678E 状压DP
    Codeforces 667C DP
    POJ 3017 DP + 单调队列 + 堆
    Codeforces 1154F (DP)
    Codeforces 1154G 枚举
    Codeforces 1153D 树形DP
    Codeforces 1109E 线段树
    Codeforces 1109C 线段树
    Codeforces 1109D (树的计数问题)
    async/await
  • 原文地址:https://www.cnblogs.com/hunterCecil/p/5630970.html
Copyright © 2011-2022 走看看