【Spring源碼解析】—— 簡單工廠模式的BeanFactory的超簡版實現

 1、什麼是簡單工廠模式

設計模式的核心是「分工」,經過分工將對象與職責劃分的更細化,進而提高系統設計的可擴展性,使其更容易維護。java

 

開閉原則:對擴展開放,對修改關閉;要增長一個新的處理邏輯,能夠開一個新的類,不要在老的上面修改spring

依賴倒轉原則:依賴關係從具體轉向抽象,也就是說:A調用B,不是直接調用B的實現,而是依賴B的接口設計模式

迪米特法則:類儘可能少的與其餘類發生關係,或者產生依賴,以此來使擴展能夠更容易dom

 

工廠模式中的三種:簡單工廠模式、工廠方法模式、抽象工廠模式;實現了建立者和調用者的分離,調用者不須要知道具體的建立者是什麼類,只須要知道工廠的接口以及本身想要的產品名稱,就能夠進行調用獲得想要的產品ide

簡單工廠模式:簡單工廠模式也稱爲靜態工廠模式,工廠類通常採用靜態方法,根據接收的參數不一樣來肯定返回對象實例,但簡單工廠模式違反了開閉原則,要增長一個新的類別必需要修改代碼測試

注意,簡單工廠模式就是:針對一個項目或者一個獨立模塊只有一個工廠類,而工廠方法模式是有一組實現了相同接口的工廠類(雖然符合開閉原則,可是會增長新的類來擴展,看狀況而定,實際上在項目開發中一般仍是用簡單工廠比較多)this

2、依據Spring中的BeanFactory本身實現簡版

首先是,先寫一個接口類,BeanFactory的接口類以下:spa

public interface BeanFactory {
    Object getBean(String beanName);
}.net

下面是xml配置文件 springtest.xml:設計

<?xml version="1.0" encoding="UTF-8"?>

<beans>

    <bean id="usertest" class="beanfactory.demo.User">

        <property name="username" value="lxlx" />

        <property name="passWord" value="111" />

        <property name="age" value="11"/>

    </bean>

</beans>

下面是bean定義的class文件 User類:

public class User {
    private String username;
    private String passWord;
    private int age;

    public void setUsername(String username) {
        this.username = username;
    }
public void setPassWord(String passWord) { this.passWord = passWord; } public void setAge(int age) { this.age = age; } public String getUsername() { return username; } public String getPassWord() { return passWord; } public int getAge() { return age; } }

接下來是實現類 ConcreteBeanFactory:

package beanfactory.demo;


import org.dom4j.io.SAXReader;

import org.dom4j.*;

import java.io.File;

import java.lang.reflect.Type;

import java.util.HashMap;

import java.util.Map;

import java.util.Iterator;

import java.lang.reflect.Method;

/**

 * Created by xiami on 2019/5/26.

 */

public class ConcreteBeanFactory implements BeanFactory{


    private Map<String, Object> beanDefinitionMap = new HashMap<String, Object>();


    //簡單工廠模式的特徵是:一個工廠中能夠生產多種不一樣的產品,這裏的Bean實際上是沒有區分不一樣的bean,是能夠經過get返回不一樣的bean

    @Override

    public Object getBean(String beanName) {

        return beanDefinitionMap.get(beanName);

    }


    //增長一個init的操做方法

    //從xml配置文件中進行解析讀取

    public void init(String xmlPath){

        SAXReader saxReader = new SAXReader();

        File file = new File(xmlPath);

        try {

            Document document = saxReader.read(file);

            Element root = document.getRootElement();

            Element foo;

            // 遍歷bean

            for (Iterator i = root.elementIterator("bean"); i.hasNext();) {

                foo = (Element) i.next();

                // 獲取bean的屬性id和class

                Attribute id = foo.attribute("id");

                Attribute cls = foo.attribute("class");

                // 利用Java反射機制,經過class的名稱獲取Class對象

                Class<?> bean = Class.forName(cls.getText());

                // 獲取對應class的信息

                java.beans.BeanInfo info = java.beans.Introspector.getBeanInfo(bean);

                // 獲取其屬性描述

                java.beans.PropertyDescriptor pd[] = info.getPropertyDescriptors();

                // 設置值的方法

                Method mSet = null;

                // 建立一個對象

                Object obj = bean.newInstance();

                // 遍歷該bean的property屬性

                for (Iterator ite = foo.elementIterator("property"); ite.hasNext();) {

                    Element foo2 = (Element) ite.next();

                    // 獲取該property的name屬性

                    Attribute name = foo2.attribute("name");

                    String value = null;

                    Object typeValue = null;

                    //獲取value值

                    value = foo2.attributeValue("value");

                    for (int k = 0; k < pd.length; k++) {

                        if (pd[k].getName().equalsIgnoreCase(name.getText())) {

                            mSet = pd[k].getWriteMethod();

                            //設置值這裏,須要根據類型給value作類型轉換

                            //properties中包含了properType的項,由於當前程序中就只有String和Int,先處理這樣的類型

                            Type mType = pd[k].getPropertyType();

                            if (mType.getTypeName().equals("java.lang.String")){

                                typeValue = String.valueOf(value);

                            }

                            else if(mType.getTypeName().equals("int")){

                                typeValue = Integer.parseInt(value);

                            }

                            mSet.invoke(obj, typeValue);

                        }

                    }

                }

                // 將對象放入beanMap中,其中key爲id值,value爲對象

                beanDefinitionMap.put(id.getText(), obj);

            }

        }catch (Exception e){

            System.out.println(e.toString());

        }

    }

}

 

下面是測試類:

public class Client {

    public static void main(String[] args){

        AbstractBeanFactory absbf = new AbstractBeanFactory();

        absbf.init("E:\\java-demo\\src\\beanfactory\\demo\\springtest.xml");

        User user = (User)absbf.getBean("usertest");

        System.out.println("User類的bean有沒有建立成功:" + user);

        System.out.println("屬性值:" + user.getUsername() + "," + user.getPassWord() + "," + user.getAge());

    }

}

測試結果是:

 

要理解的是:簡單工廠模式是一種思想,就是:不針對特定的產品進行工廠的劃分,也就是說沒有多個批次或者類別的工廠,而是全部的內容都在一個工廠裏面生產,你須要什麼我給你什麼便可

 

參考文章:https://blog.csdn.net/mlc1218559742/article/details/52776160/

相關文章
相關標籤/搜索