读《Effective Java》笔记之 第一条:考虑用静态工厂方法代替构造器

静态工厂方法与构造器相比的优势:

  1. 它们有名称:可以更好地指示出使用这一方法创建出的对象所具有的特征。(尤其是当多个构造器只有参数顺序不同的时候)
  2. 不必在每次调用的时候,都创建一个新的对象。这使得不可变类可以使用预先构建好的实例,或者作为缓存,重复利用。(例如 Boolean.valueOf(boolean))有助于控制类的实例(实例受控的类 instance-controlled)。
  3. 可以返回原返回类型的任何子类型的对象,也可以随参数不同而返回不同的子类型。这种用法常用于基于接口的框架。例如Collection接口和与之对应的不可实例化类Collections。
  4. 在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();

静态工厂方法的缺点:

  1. 类如果不含public或者protected的构造器,就不能被子类化。不过这也可能是好事,因为它鼓励使用复合,而不是继承。
  2. 第二个缺点是,静态工厂方法和其它普通静态方法实际上没有任何区别。在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();
    }
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值