01-考虑用静态工厂方法代替构造器

来源:互联网 发布:每日龙虎榜数据 编辑:程序博客网 时间:2024/06/05 04:01

本文目录

  1. 简介
  2. 优势
  3. 缺点
  4. 服务提供者框架

简介

如果想要获取一个类的实例,最常用的方法就是提供一个公有的构造器,除此之外,类还可以提供一个公有的静态工厂方法,它是一个返回类的实例的静态方法。

下面是Java中自带的Boolean类(基本类型boolean的包装类)中的静态工厂方法,这个方法将基本类型boolean值转换成一个Boolean对象引用:

 public static Boolean valueOf(boolean b) {     return (b ? TRUE : FALSE); }

我们有两种方式获取Boolean的实例:

//使用构造器Boolean boolean1=new Boolean(true);//使用静态工厂方法Boolean boolean2=Boolean.valueOf(true);

同样是传入true作为参数,两种方式有何差异?


优势

1.静态工厂方法拥有名称。
一个类只能有一个带有指定签名的构造器。通过改变参数列表中参数的顺序,达到有多个构造函数的目的,显然这不是一个好主意。相比之下,静态工厂方法没有这个限制,对于指定签名,可以通过取不同的名称表示不同的构造方式,因此静态工厂方法进行对象实例化更加灵活。

2.不用每次调用静态工厂方法时创建一个新对象。
如果程序经常请求创建相同的对象,并且创建对象的代价很高,则它可以大幅提升性能。静态工厂方法能够为重复的调用返回相同对象,这样有助于类能严格控制某个时刻哪些实例应该存在,适用于单例模式。

3.在创建参数化类型实例时,使代码变得更加简洁。

//使用构造器Map<Integer,String> map1=new HashMap<Integer,String>();//使用静态工厂方法Map<Integer,String> map2=HashMap.newInstance();

当然,使用上面的静态工厂方法,前提是HashMap类中提供了这个静态工厂方法:

public static <K,V>HashMap<K,V> newInstance(){    return new HashMap<K,V>();}

很遗憾,我看了HashMap类,至今也没有这个方法。

4.可以返回 原返回类型的任何子类型的对象。
静态工厂方法返回的对象所属的类,在编写包含该静态方法的类时可以没有,后来进行动态添加。这种灵活的方法构成了 服务提供者框架 ( Service Provider Framework)的基础,例如 JDBC API。服务提供者框架是指:多个服务提供者实现一个服务,系统为服务提供者的客户端实现多个实现,并把他们从多个实现中解耦处来。


缺点

1.如果类中不存在非私有的构造器,就不能被子类化。

2.它们与其他的静态方法实际上没有任何区别。
在API文档中,静态工厂方法并没有像构造器那样标识出来,因此,要查找是比较困难的,但是我们通过类/接口注释中关注静态工厂,并进行标准命名,也可以弥补这一缺憾,下面是一些惯用名称:

valueOf:该方法返回的实例与它的参数具有相同的值,实际上是类型转换方法,例如Boolean的;

of:这是valueOf的简洁替代。

getInstance:返回的实例是通过方法的参数来描述的,但是不能够说与参数具有相同的值。对于Singleton来说,该方法没有参数,并返回唯一的实例。

newInstance:像getInstance一样,但newInstance能够确保返回的每个实例都与所有其他实例不同。

getType:像getInstance一样,但是在工厂方法处于不同的类中的时候使用,Type表示工厂方法所返回的对象类型。

newType:像newInstance一样,但是在工厂方法处于不同的类中的时候使用,Type表示工厂方法所返回的对象类型。


服务提供者框架

这里写图片描述

以下例子模拟使用一卡通进出地铁
1.

/** * 进出地铁 服务定义接口 */public interface SubWayInterface {    // 进地铁    public boolean in();    // 出地铁    public boolean out();}

2.

/** * 一卡通进出地铁 服务具体实现类 */public class SubWayImpl implements SubWayInterface {    public boolean in() {        System.out.println("通过青岛一卡进地铁");        /**         * 是否可以进入逻辑         */        return false;    }    public boolean out() {        System.out.println("通过青岛一卡出地铁");        /**         * 是否可以放行逻辑         */        return false;    }}

3.

/** * 进出地铁 提供服务者接口 */public interface SubwayProviderInterface {    public SubWayInterface getService();}

4.

/** * 进出地铁 提供服务者实现类 */public class SubwayProviderImpl implements SubwayProviderInterface {    static {        ServiceManager.registerProvider("青岛一卡通",                 new SubwayProviderImpl());    }    public SubWayInterface getService() {        return new SubWayImpl();    }}

5.

/** * 服务提供者注册类 */public class ServiceManager {    private ServiceManager() {}    private static final Map<String, SubwayProviderInterface> providers = new ConcurrentHashMap<String, SubwayProviderInterface>();    //注册    public static void registerProvider(String name, SubwayProviderInterface p) {        providers.put(name, p);    }    //获取    public static SubWayInterface getService(String name) {        SubwayProviderInterface p = providers.get(name);        if (p == null) {            throw new IllegalArgumentException(name + "无效");        }        return p.getService();    }}

6.测试类

public class Test {    public static void main(String[] args) {        try {            Class.forName("ejava.SubwayProviderImpl");            SubWayInterface swi = ServiceManager.getService("青岛一卡通");            swi.in();            swi.out();        } catch (ClassNotFoundException e) {            e.printStackTrace();        }    }}

看看测试类的代码,有没有像极了连接数据库的代码 O(∩_∩)O ~~

Class.forName("com.mysql.jdbc.Driver");Connection conn = DriverManager.getConnection(    "jdbc:mysql://localhost:3306/student", "root", "root");conn.createStatement(); ......
0 0
原创粉丝点击