静态工厂方法与构造器相比的优势:
- 它们有名称:可以更好地指示出使用这一方法创建出的对象所具有的特征。(尤其是当多个构造器只有参数顺序不同的时候)
- 不必在每次调用的时候,都创建一个新的对象。这使得不可变类可以使用预先构建好的实例,或者作为缓存,重复利用。(例如 Boolean.valueOf(boolean))有助于控制类的实例(实例受控的类 instance-controlled)。
- 可以返回原返回类型的任何子类型的对象,也可以随参数不同而返回不同的子类型。这种用法常用于基于接口的框架。例如Collection接口和与之对应的不可实例化类Collections。
- 在JDK 1.8以前,可以利用方法的类型推断来使代码更加简洁。如下例:
// JDK 1.8以前,需要写两遍类型参数
Map<String, List<String>> m = new HashMap<String, List<String>>();
// 定义如下的静态工厂方法后
public static <K, V> HashMap<K, V> newInstance() {
return new HashMap<K, V>();
}
// 之前的创建HashMap代码可以简化成
Map<String, List<String>> m = HashMap.newInstance();
静态工厂方法的缺点:
- 类如果不含public或者protected的构造器,就不能被子类化。不过这也可能是好事,因为它鼓励使用复合,而不是继承。
- 第二个缺点是,静态工厂方法和其它普通静态方法实际上没有任何区别。在Java Doc中,它没有像构造器那样被明确标示出来。因此,对于提供了静态工厂方法而不是构造器的类来说,想查明要如何实例化这个类,是相对困难的。为弥补这一缺点,静态工厂方法有一些惯用名称:valueOf,of,getInstance,newInstance,getType,newType(Type代表具体的类型)。
其它知识点:
实例受控的类:编写实例受控的类有几个原因:1)为了实现Singleton或者是不可实例化的类。2)使得不可变的类可以确保不会存在两个想等的实例,从而使得可以用==操作符代替equals(Object)方法,可以提升性能(就像枚举类型)。
编程习惯:使用类似Collections这样的静态工厂方法时,通常会通过接口来引用被返回的对象,而不是通过具体的实现类来引用被返回的对象,这是一种良好的习惯。
服务提供者框架:在编写静态工厂方法这个类时,他所返回的对象所属的类可以不必存在(例如JDBC API)。这种灵活性构成了服务提供者框架(Service Provider Framework)的基础。SPF是指这样一个系统:多个服务提供者实现同一个服务,SPF系统为客户端提供这些(多个)实现,并把客户端从多个实现中解耦出来。
服务提供者框架有三个重要的组件:服务接口(Service Interface),这是需要提供者实现的;提供者组册API(Provider Registration API),这是系统用来注册实现,让客户端能够访问这些实现的;服务访问API(Service Access API),是客户端用来获取服务的实例的。SPF还有第四个可选组件:服务提供者接口(Service Provider Interface),是SPF系统用来创建服务提供者实例的。如果没有SPI,那SPF系统就按照类名注册,通过放射来进行实例化。
一个SPF框架的简单例子:
// Service provider framework sketch
// Service interface
public interface Service {
... // Service-specific methods go
here
}
// Service provider interface
public interface Provider {
Service newService();
}
// Noninstantiable class for service registration and access
public class Services {
private Services() { } // Prevents instantiation (Enforce n)
// Maps service names to services
private static final Map<String, Provider> providers =
new ConcurrentHashMap<String, Provider>();
public static final String DEFAULT_PROVIDER_NAME = "<def>";
// Provider registration API
public static void registerDefaultProvider(Provider p) {
registerProvider(DEFAULT_PROVIDER_NAME, p);
}
public static void registerProvider(String name, Provider p){
providers.put(name, p);
}
// Service access API
public static Service newInstance() {
return newInstance(DEFAULT_PROVIDER_NAME);
}
public static Service newInstance(String name) {
Provider p = providers.get(name);
if (p == null)
throw new IllegalArgumentException(
"No provider registered with name: " + name);
return p.newService();
}
}