《Effective Java》笔记
Java语言支持四种类型:接口、类、数组、基本类型
类的成员由它的域field、方法、成员类、成员接口组成。
方法的签名signature由它的名称和所有参数类型组成,签名不包括它的返回类型。
API元素:类、接口、构造器、成员、序列化形式。
第二章 创建和销毁对象 7条
第1条 用静态工厂方法替代构造器
1 public static Boolean valueOf(boolean b){ 2 3 return b?Boolean.TRUE : Boolean.False; 4 5 }
这样的好处是
1、调用的时候有方法名,可以提供更多的信息。
比存在多个入参一样的构造函数的时,用入参顺序互相区分好的多。
1 BigInteger(int,int Random); 2 3 BigInteger.makePrime(int,int Random);//多方法,可以有更多说明
2、不用再每次调用的时候都创建一个新对象。
可以让不可变类使用预先构建好的实例,或者缓存实例重复利用。
可以控制每个类都是Singleton或者不可实例化的。这样就可以用a==b代替a.equals(b)来提高性能。
3.可以返回原返回类型的任何子类型的对象,在选择返回类时有了更大的灵活性。
API可以返回对象,又不会让对象的类变成公有的。使得API变得简洁。私有的类不显示,使用public的接口调用 private static的实现。同时接口的实现还能不断的变化,对调用者透明。如:
1 public static <T> int binarySearch(List<? extends T> list, T key, Comparator<? super T> c) { 2 if (c==null) 3 return binarySearch((List<? extends Comparable<? super T>>) list, key); 4 5 if (list instanceof RandomAccess || list.size()<BINARYSEARCH_THRESHOLD) 6 return Collections.indexedBinarySearch(list, key, c); 7 else 8 return Collections.iteratorBinarySearch(list, key, c); 9 } 10 11 private static <T> int indexedBinarySearch(List<? extends T> l, T key, Comparator<? super T> c) {
4、创建参数化类型实例,是的代码更简洁 (现在关于泛型的已经改进)
1 Map<String,List<String>> m = new HashMap<String,List<String>>() 2 3 //改进为 4 5 public static<K,V> HashMap<K,V> newInstance(){ 6 7 return new HashMap<K,V>(); 8 9 }
第2条 遇到多个构造器参数时要用构建器
当一个类的构造方法中需要传入许多参数,且除去几个必须的,还有很多可选的参数,可以用构建器来构建对象。这个在Google的DataStore中也比较常见。
1 //Builder Pattern 2 public class NutritionFacts{ 3 private final int servingSize; 4 private final int servings; 5 private final int fat; 6 private final int sodium; 7 8 public static class Builder{ 9 // Required param 10 private final int servings; 11 private final int servingSize; 12 13 //Optional parameters - initialized to default values 14 private int fat = 0; 15 private int sodium = 0; 16 17 public Builder(int servingSize, int servings){ 18 this.servingSize = servingSize; 19 this.servings = servings; 20 } 21 public Builder fat(int val){ 22 fat = val; 23 return this; 24 } 25 public Builder sodium(int val){ 26 sodium = val; 27 return this; 28 } 29 public NutritionFacts build(){ 30 return new NutritionFacts(this); 31 } 32 } 33 34 private NutritionFacts(Builder builder){ 35 servingSize = builder.servingSize; 36 servings = builder.servings; 37 fat = builder.fat; 38 sodium = builder.sodium; 39 } 40 }
调用:
1 NutritionFacts cocaCola = new NutritionFacts.Builder(240, 8).calories(100).sodium(35).build();
第3条 用私有构造器或者枚举类型强化Singleton属性
可以搜素创建singleton实例的7中方法。
transient-瞬态:被瞬态定义的变量不可序列化。变量只会存储在内存中,不会被序列化作为本地缓存或者网络传输。
1 enum EnumSingleton{ 2 INSTANCE; 3 public void doSomeThing(){ 4 } 5 }
这种方法无偿的提供了序列化机制,绝对防止了多次实例化,即使是面对复杂的序列化或者反射攻击的时候。
1 public enum FruitEnum { 2 APPLE, ORAGE 3 }
Java在创建枚举时会创建一个继承自java.lang.Enum的类,创建两个FruitEnum对象,同时分别赋值给APPLE和ORANGE这两个属性,调用的构造函数是定义在 java.lang.Enum中的protected Enum(String name, int ordinal)方法。在创建完成两个FruitEnum对象并且分别赋值给APPLE和ORIGIN之后,还创建了一个名叫ENUM$VALUES的数组,然后把APPLE和ORIGIN按照定义的顺序放如这个数组中。
除了这个静态初始化器之外,编译器还为我们生成了两个静态方法,values()和 valueOf(java.lang.String)方法。其中values()方法将ENUM$VALUES数组拷贝一份然后返回,而valueOf(java.lang.String)方法则会调用java.lang.Enum类中的valueOf方法,其作用是根据参数名找到对应的具体的枚举对象,如果找不到的话会抛出一个IllegalArgumentException异常。
从上面的叙述可以看到,我们定义的枚举类型,经过编译器的处理最终会编程一个对象的定义,其中的枚举变量其实就是类的静态变量,因此Java中的枚举类型其实是具有很多对象的特性的,只不过平时我们都不太用到,比如枚举可以实现接口(不能继承)、定义方法等等。为了保证枚举类型像Java规范中所说的那样,每一个枚举类型极其定义的枚举变量在JVM中都是唯一的,在枚举类型的序列化和反序列化上,Java做了特殊的规定。
在序列化的时候Java仅仅是将枚举对象的name属性输出到结果中,反序列化的时候则是通过java.lang.Enum的valueOf方法来根据名字查找枚举对象。同时,编译器是不允许任何对这种序列化机制的定制的,因此禁用了writeObject、readObject、readObjectNoData、writeReplace和readResolve等方法。
http://mysun.iteye.com/blog/1581119
第4条 通过私有构造器强化不可实例化的能力
将构造函数私有化,这样没有默认构造函数,同时应为子类的构造器都必须显示或者隐式调用父类。所以这样的类不能实例化也不能被继承。
第5条 避免创建不必要的对象