何时以及如何创建对象,何时以及如何避免创建对象,如何确保适时的销毁,以及如何在对象销毁之前进行必要的清理动作
第1条、用静态工厂方法代替构造器
类中提供一个公有的静态工厂方法(static factory method
),只返回类的实例。
如来自 Boolean 的简单示例
public static final Boolean TRUE = new Boolean(true);
public static final Boolean FALSE = new Boolean(false);
@HotSpotIntrinsicCandidate
public static Boolean valueOf(boolean b) {
return (b ? TRUE : FALSE);
}
1、可以拥有名称
当一个类需要多个带有相同方法签名的构造器时,就用静态工厂方法代替构造器,并且仔细选择名称
2、不必在每次调用都创建新对象
不可变类可以使用预先构建好的实例,进行重复利用,避免创建不必要的重复对象。
如 Boolean.valueOf(boolean)
方法,不创建对象,类似享元模式。如果程序要求创建相同对象,并且创建对象的代价很高,这种方法可以极大的提升性能。
3、可以返回子类型的对象
???todo
4、根据方法的参数值,返回对象的类可以变化
如 EnumSet
没有公有的构造器,只有静态工厂方法,返回两个子类之一的实例,具体取决于底层枚举类型的大小
EnumSet(Class<E>elementType, Enum<?>[] universe) {
this.elementType = elementType;
this.universe = universe;
}
public static <E extends Enum<E>> EnumSet<E> noneOf(Class<E> elementType) {
Enum<?>[] universe = getUniverse(elementType);
if (universe == null)
throw new ClassCastException(elementType + " not an enum");
if (universe.length <= 64)
return new RegularEnumSet<>(elementType, universe);
else
return new JumboEnumSet<>(elementType, universe);
}
这两个实现类对于客户端来说是透明的,如果RegularEnumSet
不能为少量枚举类型提供性能优势,在未来发行版本中可能会删除,如果事实证明可以提升性能,那么未来可能会新增更多的EnumSet
实现类
5、方法返回对象的类,在编写当前类时可以不存在
构成了服务提供者框架的基础,如 JDBC API
,多个服务提供者实现一个服务,系统为服务提供者的客户端提供多个实现,并且把它们从多个实现中解耦出来。
服务提供者框架四个重要的组件
- 服务接口:提供者实现的,
JDBC
中的Connection
- 提供者注册API:提供者用来注册实现的,JDBC中的
DriverManager.registerDriver
- 服务访问API:客户端用来获取服务的实例。JDBC中的
DriverManager.getConnection
- 服务提供者接口:生产服务接口实例的工厂对象,JDBC中的
Driver
Java
平台提供了一个通用的服务提供者框架 java.util.ServiceLoader
6、缺点:类如果不含有公有或者受保护的构造器,就不能被子类化
但是正好符合鼓励复合而不是继承的建议