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
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值