我們都知道Spring中IOC是使用的工廠模式,但是對於實現細節就一知半解了,今天這篇文章就帶大家解讀Spring中是如何使用工廠模式的。
在上篇文章中我們懂了什麼是工廠模式,這篇文章就帶著學過的概念深入到Spring的內部看看究竟是怎麼使用該模式的。
在Spring中使用工廠方法的是BeanFactory和FactoryBean<>介面。
使用了Spring框架,我們就很少自己進行物件的建立了,而我們使用到的物件當然就是交給Spring的工廠模式來建立的了。
其中BeanFactory是Spring容器的頂層介面,也是Bean工廠最上層的介面,其會有很多工廠實現例如ClassPathXmlApplicationContext、XmlWebApplicationContext其中最常見的就是DefaultListableBeanFactory,我們可以把BeanFactory看成是一種工廠方法模式。
在工廠方法模式中有四個角色:
1、抽象工廠(Abstract Factory):提供了建立產品的介面,呼叫者通過它存取具體工廠的工廠方法 createProduct() 來建立產品。
2、具體工廠(ConcreteFactory):主要是實現抽象工廠中的抽象方法,完成具體產品的建立。
3、抽象產品(Product):定義了產品的規範,描述了產品的主要特性和功能。
4、具體產品(ConcreteProduct):實現了抽象產品角色所定義的介面,由具體工廠來建立,它同具體工廠之間一一對應。
BeanFactory是角色中的抽象工廠,而getBean就相當於我們範例中的createProduct()方法,用於建立物件。
DefaultListableBeanFactory等實現的工廠類就是角色中的具體工廠。
產品就是在我們開發中加上註解的@Controller@Service@compent@Configuration的類,Spring會將他們當成產品使用工廠模式生成物件。
在我們createProduct方法中是直接建立的物件,Spring肯定不會這麼的笨,我們接著看getBean方法的實現方法。
getBean方法中一系列鏈路最終呼叫到doGetBean方法用於建立物件,在doGetBean中建立物件使用的是設計模式中的代理模式。
講完BeanFactory,再介紹另外一個工廠模式的應用FactoryBean<>,想必大家也經常會比較這倆。
實際上,這兩個介面都是用於建立物件,都可以看做是工廠方法模式的實現。
FactoryBean<>工廠通過getObject()方法來建立並返回物件,我們可以通過實現FactoryBean<>來客製化化自己需要的Bean物件。
既然FactoryBean<>是留個開發者實現的,我們就手動實現一個FactoryBean<>。
@Component
public class MyFactoryBean<> implements FactoryBean<><Product>, BeanNameAware {
private String name;
@Override
public Product getObject() throws Exception {
return new Product();
}
@Override
public Class<?> getObjectType() {
return Product.class;
}
@Override
public boolean isSingleton() {
return false;
}
@Override
public void setBeanName(String s) {
this.name = name;
}
}
接著我們分別使用兩種方式getBean。
public static void main(String[] args) {
//測試FactoryBean<>
ApplicationContext ac =new AnnotationConfigApplicationContext(MyFactoryBean<>.class);
System.out.println(ac.getBean("myFactoryBean<>"));
System.out.println(ac.getBean("&myFactoryBean<>"));
}
只是在獲取Bean時加了一個&會出現兩種情況。
cn.sky1998.create.methodFactory.Product@52a86356
cn.sky1998.create.methodFactory.MyFactoryBean<>@5ce81285
doGetBean裡面進行了判斷是否是FactoryBean<>的實現類。
Spring對FactoryBean<>的實現機制是當你獲取一個Bean時,如果獲取的Bean的型別是FactoryBean<>,並且其name中並沒有&則呼叫bean的getObject方法獲取FactoryBean<>實現類中提供bean,否則就是直接返回普通的bean型別。
當我們在引入其他框架整合Spring時,便會有很多橋接整合包,例如mybatis-spring等,其中就會有FactoryBean<>的實現,例如SqlSessionFactoryBean<>、MapperFactoryBean<>等,將需要整合的客製化化Bean通過工廠方法的模式,加入進Spring容器中。
當我們整合Mybatis時無論是多麼的Bean都能很好的處理,mybatis-spring中的FactoryBean<>正是將Mybatis的核心啟動類給呼叫出來。
BeanFactory是Spring的一個大工廠,建立著Spring框架執行過程中所需要的Bean;
而FactoryBean<>是一個客製化化工廠,其會存在於BeanFactory建立物件的過程中,當有需要時,會通過FactoryBean<>去自客製化個性化的Bean,從而Spring框架提高擴充套件能力。
這篇文章加深工廠方法的理解的同時,又進一步的深入了BeanFactory和FactoryBean<>的實現細節。
設計模式相關部落格: