(从今天开始啃Effective Java!)
第一章 引言
跳过
第二章 创建和销毁对象
第一条 用静态工厂方法代替构造器
一 例子
public static Boolean
valueOf(boolean b
) {
return b
? Boolean
.TRUE
: Boolean
.FALSE
;
}
二 与构造器相比的优势
1 有名称
更有阅读性。 比如用BigInteger.probablePrime的静态工厂方法来取得素数明显比BigInteger(int,int,Random)更加清楚。
2 不必每次调用都创建新对象
如果程序经常请求创建相同的对象,并且创建对象的代价很高时,可以极大提升性能。 能够为重复的调用返回相同的对象。
3 可以返回原返回类型的任何子类对象
返回对象有了更大的灵活性。 比如API可以返回对象,同时又不会使对象的类变为公有的。
4 所返回的对象的类可以随着每次调用而变化
如EnumSet:如果它的元素小于等于64个,静态工厂方法会返回一个RegalarEnumSet实例,用单个long支持;如果它的元素大于64个,静态工厂方法会返回一个JumboEnumSet实例,用一个long数组支持。 客户端不需要关心他们从工厂方法中获得的类,在未来的版本中添加或者删除EnumSet的实现都不会造成任何负面影响。
5 返回对象所属的类在编写包含该静态工厂方法的类时可以不存在
(此处还不是很理解,日后回过头再看看) 这是服务者框架的基础,例如JDBC API。 服务者框架:多个服务提供者实现一个服务,系统为服务提供者的客户端提供多个实现,并把它们从多个实现中解耦出来。 服务者框架三个重要组件:服务接口(提供者实现的)、提供者注册API(提供者用来注册实现的)、服务访问API(客户端用来获取服务的实例)。 服务者框架第四个可选组件:服务提供者接口(产生服务接口实例的工厂对象)。 对于JDBC来说,Connection就是其服务接口的一部分,DriverManager.registerDriver是提供者注册API,DriverManager.getConnection是服务访问API,Driver是服务提供者接口。
三 缺点
1 类如果不含公有的或者受保护的构造器,就不能被子类化
2 程序员很难发现它们
四 静态工厂方法的一些惯用名称
from —— 类型转换方法,只有单个参数,返回该类型的一个实例
Date d
= Date
.from(instant
);
of —— 聚合方法,带有多个参数,返回该类型合并的一个实例
Set
<Rank> faceCards
= EnumSet
.of(JACK
,QUEEN
,KING
);
valueOf —— 比from和of更繁琐的一种替代方法
BigInteger prime
= BigInteger
.valueOf(Integer
.MAX_VALUE
);
instance 或者 getInstance —— 返回的实例是通过方法的参数来描述的,但是不能说与参数具有同样的值
StackWalker luke
= StackWalker
.getInstance(options
);
create 或者 newInstance —— 同getInstance,但每次调用都返回一个新的实例
Object newArray
= Array
.newInstance(classObject
,arrayLen
);
getType —— 同getInstance,但在工厂方法处于不同的类中的时候使用,Type表示工厂方法所返回的对象类型
FileStore fs
= Files
.getFileStore(path
);
newType —— 同newInstance,但在工厂方法处于不同的类中的时候使用,Type表示工厂方法所返回的对象类型
BufferedReader br
= Files
.newBufferedReader(path
);
type —— getType和newType的简版
List
<Complaint> litany
= Collections
.list(legacyLitany
);
五 总结
静态工厂经常更加合适,因此切忌第一反应就是提供公有的构造器,而不考虑静态工厂。