代码之家  ›  专栏  ›  技术社区  ›  Cheok Yan Cheng

Joshua Bloch对构建器模式的一些修改

  •  5
  • Cheok Yan Cheng  · 技术社区  · 14 年前

    我使用Joshua Bloch介绍的JavaBuilder模式。有时,我会发现某些字段与基元类型相比,用默认值初始化更昂贵。

    因此,我的策略是。

    1. 我延迟了这些字段的默认值初始化操作。
    2. 在构建期间,如果调用方以前没有设置它们,那么我将只将它们初始化为默认值。

    我不确定是不是 好的 这样做?可能会有什么收获吗?比如,线程安全问题?到目前为止,我没有看到任何问题。

    package sandbox;
    
    import java.util.Calendar;
    
    /**
     *
     * @author yccheok
     */
    // Builder Pattern
    public class NutritionFacts {
        private final int servingSize;
        private final int servings;
        private final int calories;
        private final int fat;
        private final int sodium;
        private final int carbohydrate;
    
        private final java.util.Calendar calendar;    // !!!
    
        public static class Builder {
            // Required parameters
            private final int servingSize;
            private final int servings;
            // Optional parameters - initialized to default values
            private int calories = 0;
            private int fat = 0;
            private int carbohydrate = 0;
            private int sodium = 0;
    
            // We suppose to provide a default value for calendar. However, it may
            // seem expensive. We will do it later during build.        
            private java.util.Calendar calendar = null;
    
            public Builder(int servingSize, int servings) {
                this.servingSize = servingSize;
                this.servings = servings;
            }
            public Builder calories(int val)
            { calories = val; return this; }
            public Builder fat(int val)
            { fat = val; return this; }
            public Builder carbohydrate(int val)
            { carbohydrate = val; return this; }
            public Builder sodium(int val)
            { sodium = val; return this; }
    
            public NutritionFacts build() {
                // !!!
                if (this.calendar == null) {
                    this.calendar = Calendar.getInstance();
                }            
                return new NutritionFacts(this);
            }
        }
    
        private NutritionFacts(Builder builder) {
            servingSize = builder.servingSize;
            servings = builder.servings;
            calories = builder.calories;
            fat = builder.fat;
            sodium = builder.sodium;
            carbohydrate = builder.carbohydrate;
            calendar = builder.calendar;
        }
    }
    
    3 回复  |  直到 14 年前
        1
  •  4
  •   nanda    14 年前

    如果我真的希望日历为空呢?那么你的模式就行不通了。否则,我就没问题了。

        2
  •  3
  •   NG.    14 年前

    我看不出有什么问题。构建器模式实际上是一个API,可以帮助完成某些任务。只要您遵守模式/API的原则,就可以在API的覆盖下做任何您想做的事情。我认为延迟加载昂贵的资源是可以的,只要确保您记录了您认为开发人员应该知道的任何行为。

        3
  •  2
  •   hhbarriuso    14 年前

    小心,因为日历是可变的,所以你应该防御性地把它复制到营养品构造器中。