静态工厂方法鸡肋吗?

对比静态工厂

静态工厂方法很容易让人想到设计模式的静态工厂,也叫简单工厂,作者说两者不同,我认为这两者仅是从概念上来讲有区别,从技术的角度来讲没有区别

简单来说,其区别体现在:静态工厂方法所在的类的意义是具体的,并且这个方法只创造与这个方法所在类直接相关的对象。而静态工厂的概念与静态工厂方法并不等同,静态工厂是更宽泛的概念,其实现包含了静态工厂方法,静态工厂里的静态工厂方法生产的不是工厂本身,是工厂里的产品,与其所在的类间接关联。

技术上没有区别,这两者的代码结构是一致的,且静态工厂方法里可以使用静态工厂来获得结果,静态工厂也可以使用产品的静态工厂方法。静态工厂方法的优点静态工厂也都可以有。

对比构造方法

  • 优势方面

静态工厂方法与构造方法相比,不同的地方在于:

  1. 静态工厂方法可以有自定义方法名,而构造方法必须等同于类名
  2. 静态工厂方法可以自定义返回值,而构造方法可以间接等同于只能返回自己的实例

纵观作者总结的5大好处:

  1. 可读性强,因为可以任意命名,还可以用于实现多构造方法
  2. 构造方法只要调用就会产生一个对象,但静态工厂方法返回的可以不是新创建的
  3. 可以返回当前类的任意子类对象,常用于接口
  4. 每次调用都可以根据参数返回不同的类,EnumSet的例子,会根据需要的大小返回不同的EnumSet
  5. 返回的对象所属的类,在写代码的时候可以不存在。

实际就是这两个方面,灵活的方法名和灵活的返回值带来的好处。

静态工厂方法的第一类好处是说有意义的方法名提高了代码可读性,这不是静态工厂方法独有的,是编程世界里所有的普通方法都有的好处。这启发我们在写代码的过程中无论你是面向过程也好,面向对象也好,面向接口,还是面向其它XXX也好,多写方法,多取有意义的方法名,这是提升代码可读性非常有效的方式,比很多开发只知道写注释有用多了。

第二类关于返回值的好处说了很多条,都很有用,观念上具有指导作用,但在现在的Java世界,一定有必要按照这种方法去实践吗?可能很多情况下都没有必要了。

比如第二点,其实主要应用就是一些缓存啊,单例之类的场景。就拿单例来说,使用静态工厂方法实现懒汉式单例,你还需要解决线程安全的问题,解决代码编译重排的问题,解决这个问题复杂吗?不复杂,有必要吗?可以没必要,使用Spring Bean,完全可以抛弃这么原始的做法。

后面几点好处比较常用的场景基本都是在使用接口的情况下,静态工厂方法可以返回接口不同的实现,那有了Spring这么强大的IoC框架,静态工厂方法作为学习,理解思路可以,实现业务的时候其实有更好的工具。

特别提一下第5点,这个着实理解了很久,最后我觉得作者可能是从编译class的角度在说,就是你这个方法可以返回一个编译时不存在,只出现在运行时的实现类。从这个角度出发,我想到两类场景,一类是通过反射在运行时获得真正的对象。典型的应用比如Class.forName。还有一类场景是使用动态配置的场景,比如我创造一个map,在运行里会加载DB里的动态配置,静态工厂方法可以在写代码的时候不需要这个具体的配置类存在,再扯远一些比如枚举,其实返回的枚举类也是可以不存在的,这些场景不一定典型,但我觉得也符合这个好处。

总结一下,静态工厂方法有些场景下可以提升代码可读性,可以使用,至于单例或者解耦的场景,我们有更强大的工具可以使用,使用静态工厂方法比较原始,会有很多问题需要去解决。不过,如果用不了强大的工具,那也只能回归原始。带来的启发是,多写方法,多用接口。

  • 劣势方面

静态工厂方法当然也有缺点:

  1. 使用静态工厂方法通常需要把构造方法私有化,因此不能被子类化。
  2. 静态工厂方法比较隐晦,不易被发现,一般可以采用约定的名称来命名使其容易被发现。

都是小问题。在不容易被发现这点后提了一些静态工厂方法的约定,我倒觉得有点意思。像from,of,instance这些词很有意义吗?我觉得并不,那岂不是不符合静态工厂方法可以增加可读性的特点?对,胡乱使用的情况下我觉得还不如回归到构造方法。但注意,作者实际上补充了一大堆这些命名背后的含义,也就是说这些命名需要基于约定。达成约定后这些命名含义就具体了。然而我觉得这件事不简单,你比如说Java官方的Optional.of(),并不符合作者要求的of用于聚合的情景。

那到底还用不用,是不是鸡肋呢?我支持用,原因是这样写出来的代码更美观,这可能是第6个优点:赏心悦目。

你可能感兴趣的:(静态工厂方法鸡肋吗?)