从简单工厂到Spring配置文件

让我们来看这样一段代码:

想象这样一个场景,你是一个手机测试厂商,现在你需要测试所有的Iphone手机。但是具体的测试方法不一样,比如Iphone6有了指纹识别,而Iphone4s没有。同时你需要考虑,未来还会新出Iphone8,Iphone9。

/*Iphone是一个抽象类,继承它的有Iphone5s,Iphone6,Iphone7等
  testIphone用来测试Iphone。
*/
Iphone testIphone(){
      Iphone iphone = new Iphone();
      iphone.powerOn();
      iphone.takePic();
      ...
}

这段代码真是好,体现了一个设计原则(面向接口编程而不是面向实现编程)这样的话,即使以后出来Iphone8,Iphone9也不需要修改代码。但是,聪明的你看出来上面代码的问题了吗,抽象类不可以创建对象,我们从抽象程度再往下走,具体应该这样写。

/*Iphone是一个抽象类,继承它的有Iphone5s,Iphone6,Iphone7等
  testIphone用来测试Iphone。
*/
Iphone testIphone(String type){
   Iphone iphone;
   if (type.equals("Iphone7")){
       iphone = new Iphone7();
    } else if (type.equals("Iphone8")){
        iphone = new Iphone8();
    } else if (type.equals("Iphone9")){
        iphone = new Iphone9();
    }
      iphone.powerOn();
      iphone.takePic();
      ...
}

运行倒是可以运行,但是以后如果Iphone7不在生产了,还需要去删掉相关代码,而且当新生产了Iphone10也得进行修改,这就违背了开闭原则(对修改关闭,对扩展开放)。怎么才能把前两段代码完美的结合起来呢,仔细观察,问题出在创建对象的时候,如果代码一后面的 new Iphone()部分可以自动跟着type改变就好了。

设计原则:找出代码中最可能变化的部分,把他们独立出来,不要和那些不变化的代码混在一起。

有了,我们把判断的那一部分拿出来封装成一个类,这样testIphone根本不需要知道到底是哪个Iphone,创建对象与使用对象完全解耦。


public Class SimpleIphoneFactory {
   public static Iphone createIphone(String type){
      Iphone iphone = null;
      if (type.equals("Iphone7")){
          iphone = new Iphone7();
       } else if (type.equals("Iphone8")){
          iphone = new Iphone8();
       } else if (type.equals("Iphone9")){
          iphone = new Iphone9();
    }
    return iphone;
  }
} 

/*Iphone是一个抽象类,继承它的有Iphone5s,Iphone6,Iphone7等
  testIphone用来测试Iphone。
*/
public Class TestIphone {
...
Iphone testIphone(String type){
      Iphone = SimpleIphoneFactory.createIphone(type);
      iphone.powerOn();
      iphone.takePic();
      ...
  }
}

通过创建工厂类来将对象的创建与对象的使用解耦,但你会说,这样依旧违背了开闭原则啊,当改变的时候,还是得在工厂类里面进行修改。
没错,这时候我们继续观察,String type传来的值其实可以与类名是一样的,如果我们在工厂类里把创建实例的部分写成创建传来的相应字符串岂不是可以?我们想到了利用反射机制

iphone = (Iphone)Class.forName(type).newInstance();

我们在想,在代码中我们可以不需要修改type来改变,可以去读取一个配置文件,这样做的好处是当bean进行修改时不用重新编译只需要更改配置文件就可以,聪明的你是否想到了Spring呢?没错,Spring里的bean.xml就相当于是配置文件啦,而ApplicationContext就相当于是工厂。

ApplicationContext ctx = new FileSystemXmlApplicationContext("bean.xml");   
Iphone iphone= null;   
iphone= (Iphone) ctx.getBean("Iphone7");  

简单工厂模式中,我们经常应用到的就是获得数据库的连接啦,数据库是抽象类,具体的mysql等是实体类。DriverManager是工厂类。通过DriverManager的getConnection就可以获得相应的连接了。

    //getConnection源码
    public static Connection getConnection(String url,
        String user, String password) throws SQLException {
        java.util.Properties info = new java.util.Properties();

        if (user != null) {
            info.put("user", user);
        }
        if (password != null) {
            info.put("password", password);
        }

        return (getConnection(url, info, Reflection.getCallerClass()));
    }
微信公众号-码农放个假
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
【社区内容提示】社区部分内容疑似由AI辅助生成,浏览时请结合常识与多方信息审慎甄别。
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

相关阅读更多精彩内容

  • 1. Java基础部分 基础部分的顺序:基本语法,类相关的语法,内部类的语法,继承相关的语法,异常的语法,线程的语...
    子非鱼_t_阅读 34,596评论 18 399
  • 一、设计模式的分类 总体来说设计模式分为三大类: 创建型模式,共五种:工厂方法模式、抽象工厂模式、单例模式、建造者...
    RamboLI阅读 825评论 0 1
  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 178,508评论 25 709
  • 1 场景问题# 大家都知道,在Java应用开发中,要“面向接口编程”。那么什么是接口?接口有什么作用?接口如何使用...
    七寸知架构阅读 6,642评论 14 70
  • 自从开始做互联网运营,看手机看微信成了工作的一部分,手机一响就忍不住去看,其实知道这样不好,很浪费时间,但就是控制...
    知识e站阅读 810评论 0 0

友情链接更多精彩内容