使用静态工厂方法代替构造函数

技术公众号:Java In Mind(Java_In_Mind),欢迎关注!

前言

我相信大部分都看过《Effective Java》这本Java神书,这本书在我学习Java的路上给了我帮助可以说是受益匪浅,书中第一篇就建议:考虑用静态工厂方法代替构造器。我自从看了之后就开始使用实践该建议,到现在已经基本偏爱静态工厂方法,这里我就谈谈使用以来的一些领悟。

使用静态工厂方法的优缺点

优点
  • 静态工厂方法有名称
  • 不必每次调用都创建一个新对象
  • 可以返回原类型的任何子类型的对象
  • 创建参数化类型实例的时候,它们使代码更加简洁
缺点
  • 类如果不含公有的或者受保护的构造器,就不能被子类化
  • 静态工厂方法与其他静态方法没有任何区别

静态工厂方法有名称

这点确实是很好地优势,静态工厂方法有自己的名称,也就意味着可以表达某种意思,或者某些特殊的实例,例如,在使用一些通用返回结果的时候,可以用类似如此的静态工厂方法来创建实例:

public class Result {
        private boolean success;
    private String message;
    private Object content;
  
    private Result(boolean success,String message,Object content){
            this.success = success;
            this.message = message;
            this.content = content;
    }
        
    public static Result ofSuccess(Object content){
            return new Result(true,"处理成功",content);
    }
    
    public static Result ofFail(String message){
      return new Result(false,message,null);
    }
  
    // getter and setter...
}
//使用
Result.ofSuccess(content);
Result.ofFail("处理失败,找不到该记录");

这样子,通过静态工厂方法就可以很好表达要构造的对象的目的,而调用方可以比较方便获取实例。

不必每次调用都创建一个新对象

这个其实很常见,如Boolean类中的几个valueOf静态工厂方法,Boolean有两个静态域:TRUEFALSE,这样就不用每次创建一个新对象:

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

其实,我们在使用单例的时候,通常获取单例的方法就是一个静态工厂方法,并且也是不用创建一个新的对象:

public class SingleTon {
    private static SingleTon singleTon = new SingleTon();
    
    private SingleTon() {
    }

    public static SingleTon getInstance() {
        return singleTon;
    }
}

可以返回原类型的任何子类型的对象

静态工厂方法返回的实例可以是该类型的子类,而这个子类可以在编写静态工厂方法时不存在,这也就意味着可以由其他方来实现,也意味着对拓展友好,还可以根据静态工厂方法的入参来选择不同的实现,非常灵活。

  • 在JDK中有个例子就是EnumSet,根据入参的枚举的枚举值量来决定使用哪种枚举实例。

  • 在使用JDBC的时候,DriveManager在获取连接前得先加载对应的实现驱动,这样就能把实现交给第三方来完成,自己只需提供上层的抽象框架

  • 日常编程中可以用于同一需求的不同实现,通过静态工厂方法选择实现,并且有利于后期拓展,例如事件处理:

    public abstract class EventHandler {
          private static Map<String,EventHandler> handlers = new HashMap<>();
        //事件类型
          public abstract String type();
          //子类注册入口    
          public static void register(EventHandler eh){
              handlers.put(eh.type(),eh);
        }
          //获取不同的实现
          public static EventHandler getHandler(Event e){
              return handlers.get(e.getType())
        }
    }
    

创建参数化类型实例的时候,它们使代码更加简洁

静态工厂方法还有一个好处就是提高代码的简洁度,也使得代码更加易读:

  • 使用JDK的类型推断,不过这在现在的JDK已经可以直接推断了

    public class Maps {
        public static <K, V> HashMap<K, V> newHashMap() {
            return new HashMap<K, V>();
        }
    }
    //使用静态工厂方法
    Map<String,Object> map = Maps.newHashMap();
    //使用构造方法
    Map<String,Object> map = new HashMap<>();//JDK1.6之前不支持这种写法
    
  • 正如第一点的例子,静态工厂方法可以有自己的名称,可以有一些默认值设置,可以减少构建实例的参数,自然也会提高代码的简洁度

类如果不含公有的或者受保护的构造器,就不能被子类化

类不含有public或者protect的构造方法时,子类的构建需要使用到父类的构造器,这个时候静态工厂方法就不适合

静态工厂方法与其他静态方法没有任何区别

使用静态工厂方法的时候,如果该类包含其他的静态方法,那么这个时候对于开发编程是不友好的,特别是在方法比较多,命名比较混乱的时候,使用者不能一眼就知道该用哪个方法来构造自己想要的实例,所以在编写静态工厂方法的时候尽量使用大家习惯的、约定俗成的命名风格:

  • of
  • valueOf
  • getInstance / newInstance
  • getType / newType,其中Type为类型,如,EventHandler.getEventHandler()

总结

结合我个人的经验,我认为使用静态工厂方法的利大于弊:

  • 提高代码的可阅读性
  • 更加符合人类的正常思维方式,想要某个类的实例直接找某个类就能获取到,而不是通过new关键字
  • 能够根据不同的需求创建不同的实例,而不用让看代码的人推测new出来的对象然后set某些值去猜测某种意图,通常我们的做法是注释,但是使用静态工厂的方法命名不是更加简洁吗
  • 提高代码的拓展性 ,方便拓展,对修改友好,符合开闭原则
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,444评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,421评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,036评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,363评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,460评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,502评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,511评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,280评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,736评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,014评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,190评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,848评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,531评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,159评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,411评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,067评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,078评论 2 352