自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

設(shè)計(jì)與實(shí)現(xiàn)資源加載器,從Spring.xml解析和注冊(cè)Bean對(duì)象

開(kāi)發(fā) 架構(gòu)
在完成 Spring 的框架雛形后,現(xiàn)在我們可以通過(guò)單元測(cè)試進(jìn)行手動(dòng)操作 Bean 對(duì)象的定義、注冊(cè)和屬性填充,以及最終獲取對(duì)象調(diào)用方法。

[[404869]]

本文轉(zhuǎn)載自微信公眾號(hào)「bugstack蟲洞?!?,作者小傅哥。轉(zhuǎn)載本文請(qǐng)聯(lián)系bugstack蟲洞棧公眾號(hào)。   

目錄

  • 一、前言
  • 二、目標(biāo)
  • 三、設(shè)計(jì)
  • 四、實(shí)現(xiàn)
    • 1. 工程結(jié)構(gòu)
    • 2. 資源加載接口定義和實(shí)現(xiàn)
    • 3. 包裝資源加載器
    • 4. Bean定義讀取接口
    • 5. Bean定義抽象類實(shí)現(xiàn)
    • 6. 解析XML處理Bean注冊(cè)
  • 五、測(cè)試
    • 1. 事先準(zhǔn)備
    • 2. 配置文件
    • 3. 單元測(cè)試(資源加載)
    • 4. 單元測(cè)試(配置文件注冊(cè)Bean)
  • 六、總結(jié)
  • 七、系列推薦

一、前言

你寫的代碼,能接的住產(chǎn)品加需求嗎?

接,是能接的,接幾次也行,哪怕就一個(gè)類一片的 if...else 也可以!但接完成什么樣可就不一定了,會(huì)不會(huì)出事故也不是能控制住的。

那出事故時(shí),你說(shuō)因?yàn)槲覍?if...else 多了導(dǎo)致代碼爛了,但可是你先動(dòng)的手?。耗阏f(shuō)的需求還得加、你說(shuō)的老板讓上線、你說(shuō)的合同都簽了,搬磚碼農(nóng)的我沒(méi)辦法,才以堆代碼平需求,需求太多不好搞,我才以搬磚平需求!諸侯不服,我才以兵服諸侯,你不服,我就打到你服!

但代碼爛了有時(shí)候并不是因?yàn)樾枨蠹拥目臁⒁膊皇侵鄙暇€。因?yàn)橥诔薪赢a(chǎn)品需求的前幾次,一個(gè)功能邏輯的設(shè)計(jì)并不會(huì)太復(fù)雜,也不會(huì)有多急迫,甚至?xí)舫鲎屇阕鲈O(shè)計(jì)、做評(píng)審、做開(kāi)發(fā)的時(shí)間,如果這個(gè)時(shí)候仍不能把以后可能會(huì)發(fā)生的事情評(píng)估到需求里,那么導(dǎo)致代碼的混亂從一開(kāi)始就已經(jīng)埋下了,以后只能越來(lái)越亂!

承接需求并能把它做好,這來(lái)自于對(duì)需求的理解,產(chǎn)品場(chǎng)景開(kāi)發(fā)的經(jīng)驗(yàn)以及對(duì)代碼實(shí)踐落地的把控能力等綜合多方面因素的結(jié)果。就像你現(xiàn)在做的開(kāi)發(fā)中,你的代碼有哪些是經(jīng)常變化的,有哪些是固定通用的,有哪些是負(fù)責(zé)邏輯拼裝的、有哪些是來(lái)做核心實(shí)現(xiàn)的。那么現(xiàn)在如果你的核心共用層做了頻繁變化的業(yè)務(wù)層包裝,那么肯定的說(shuō),你的代碼即將越來(lái)越亂,甚至可能埋下事故的風(fēng)險(xiǎn)!

在我們實(shí)現(xiàn)的 Spring 框架中,每一個(gè)章節(jié)都會(huì)結(jié)合上一章節(jié)繼續(xù)擴(kuò)展功能,就像每一次產(chǎn)品都在加需求一樣,那么在學(xué)習(xí)的過(guò)程中可以承上啟下的對(duì)照和參考,看看每一個(gè)模塊的添加都是用什么邏輯和技術(shù)細(xì)節(jié)實(shí)現(xiàn)的。這些內(nèi)容的學(xué)習(xí),會(huì)非常有利于你以后在設(shè)計(jì)和實(shí)現(xiàn),自己承接產(chǎn)品需求時(shí)做的具體開(kāi)發(fā),代碼的質(zhì)量也會(huì)越來(lái)越高,越來(lái)越有擴(kuò)展性和可維護(hù)性。

二、目標(biāo)

在完成 Spring 的框架雛形后,現(xiàn)在我們可以通過(guò)單元測(cè)試進(jìn)行手動(dòng)操作 Bean 對(duì)象的定義、注冊(cè)和屬性填充,以及最終獲取對(duì)象調(diào)用方法。但這里會(huì)有一個(gè)問(wèn)題,就是如果實(shí)際使用這個(gè) Spring 框架,是不太可能讓用戶通過(guò)手動(dòng)方式創(chuàng)建的,而是最好能通過(guò)配置文件的方式簡(jiǎn)化創(chuàng)建過(guò)程。需要完成如下操作:

如圖中我們需要把步驟:2、3、4整合到Spring框架中,通過(guò) Spring 配置文件的方式將 Bean 對(duì)象實(shí)例化。

接下來(lái)我們就需要在現(xiàn)有的 Spring 框架中,添加能解決 Spring 配置的讀取、解析、注冊(cè)Bean的操作。

三、設(shè)計(jì)

依照本章節(jié)的需求背景,我們需要在現(xiàn)有的 Spring 框架雛形中添加一個(gè)資源解析器,也就是能讀取classpath、本地文件和云文件的配置內(nèi)容。這些配置內(nèi)容就是像使用 Spring 時(shí)配置的 Spring.xml 一樣,里面會(huì)包括 Bean 對(duì)象的描述和屬性信息。在讀取配置文件信息后,接下來(lái)就是對(duì)配置文件中的 Bean 描述信息解析后進(jìn)行注冊(cè)操作,把 Bean 對(duì)象注冊(cè)到 Spring 容器中。整體設(shè)計(jì)結(jié)構(gòu)如下圖:

資源加載器屬于相對(duì)獨(dú)立的部分,它位于 Spring 框架核心包下的IO實(shí)現(xiàn)內(nèi)容,主要用于處理Class、本地和云環(huán)境中的文件信息。

當(dāng)資源可以加載后,接下來(lái)就是解析和注冊(cè) Bean 到 Spring 中的操作,這部分實(shí)現(xiàn)需要和 DefaultListableBeanFactory 核心類結(jié)合起來(lái),因?yàn)槟闼械慕馕龊蟮淖?cè)動(dòng)作,都會(huì)把 Bean 定義信息放入到這個(gè)類中。

那么在實(shí)現(xiàn)的時(shí)候就設(shè)計(jì)好接口的實(shí)現(xiàn)層級(jí)關(guān)系,包括我們需要定義出 Bean 定義的讀取接口 BeanDefinitionReader 以及做好對(duì)應(yīng)的實(shí)現(xiàn)類,在實(shí)現(xiàn)類中完成對(duì) Bean 對(duì)象的解析和注冊(cè)。

四、實(shí)現(xiàn)

1. 工程結(jié)構(gòu)

  1. small-spring-step-05 
  2. └── src 
  3.     ├── main 
  4.     │   └── java 
  5.     │       └── cn.bugstack.springframework   
  6.     │           ├── beans 
  7.     │           │   ├── factory 
  8.     │           │   │   ├── factory 
  9.     │           │   │   │   ├── AutowireCapableBeanFactory.java 
  10.     │           │   │   │   ├── BeanDefinition.java 
  11.     │           │   │   │   ├── BeanReference.java 
  12.     │           │   │   │   ├── ConfigurableBeanFactory.java 
  13.     │           │   │   │   └── SingletonBeanRegistry.java 
  14.     │           │   │   ├── support 
  15.     │           │   │   │   ├── AbstractAutowireCapableBeanFactory.java 
  16.     │           │   │   │   ├── AbstractBeanDefinitionReader.java 
  17.     │           │   │   │   ├── AbstractBeanFactory.java 
  18.     │           │   │   │   ├── BeanDefinitionReader.java 
  19.     │           │   │   │   ├── BeanDefinitionRegistry.java 
  20.     │           │   │   │   ├── CglibSubclassingInstantiationStrategy.java 
  21.     │           │   │   │   ├── DefaultListableBeanFactory.java 
  22.     │           │   │   │   ├── DefaultSingletonBeanRegistry.java 
  23.     │           │   │   │   ├── InstantiationStrategy.java 
  24.     │           │   │   │   └── SimpleInstantiationStrategy.java   
  25.     │           │   │   ├── support 
  26.     │           │   │   │   └── XmlBeanDefinitionReader.java 
  27.     │           │   │   ├── BeanFactory.java 
  28.     │           │   │   ├── ConfigurableListableBeanFactory.java 
  29.     │           │   │   ├── HierarchicalBeanFactory.java 
  30.     │           │   │   └── ListableBeanFactory.java 
  31.     │           │   ├── BeansException.java 
  32.     │           │   ├── PropertyValue.java 
  33.     │           │   └── PropertyValues.java  
  34.     │           ├── core.io 
  35.     │           │   ├── ClassPathResource.java  
  36.     │           │   ├── DefaultResourceLoader.java  
  37.     │           │   ├── FileSystemResource.java  
  38.     │           │   ├── Resource.java  
  39.     │           │   ├── ResourceLoader.java  
  40.     │           │   └── UrlResource.java 
  41.     │           └── utils 
  42.     │               └── ClassUtils.java 
  43.     └── test 
  44.         └── java 
  45.             └── cn.bugstack.springframework.test 
  46.                 ├── bean 
  47.                 │   ├── UserDao.java 
  48.                 │   └── UserService.java 
  49.                 └── ApiTest.java 

工程源碼:公眾號(hào)「bugstack蟲洞?!梗貜?fù):Spring 專欄,獲取完整源碼

Spring Bean 容器資源加載和使用類關(guān)系,如圖 6-3

圖 6-3

  • 本章節(jié)為了能把 Bean 的定義、注冊(cè)和初始化交給 Spring.xml 配置化處理,那么就需要實(shí)現(xiàn)兩大塊內(nèi)容,分別是:資源加載器、xml資源處理類,實(shí)現(xiàn)過(guò)程主要以對(duì)接口 Resource、ResourceLoader 的實(shí)現(xiàn),而另外 BeanDefinitionReader 接口則是對(duì)資源的具體使用,將配置信息注冊(cè)到 Spring 容器中去。
  • 在 Resource 的資源加載器的實(shí)現(xiàn)中包括了,ClassPath、系統(tǒng)文件、云配置文件,這三部分與 Spring 源碼中的設(shè)計(jì)和實(shí)現(xiàn)保持一致,最終在 DefaultResourceLoader 中做具體的調(diào)用。
  • 接口:BeanDefinitionReader、抽象類:AbstractBeanDefinitionReader、實(shí)現(xiàn)類:XmlBeanDefinitionReader,這三部分內(nèi)容主要是合理清晰的處理了資源讀取后的注冊(cè) Bean 容器操作。接口管定義,抽象類處理非接口功能外的注冊(cè)Bean組件填充,最終實(shí)現(xiàn)類即可只關(guān)心具體的業(yè)務(wù)實(shí)現(xiàn)

另外本章節(jié)還參考 Spring 源碼,做了相應(yīng)接口的集成和實(shí)現(xiàn)的關(guān)系,雖然這些接口目前還并沒(méi)有太大的作用,但隨著框架的逐步完善,它們也會(huì)發(fā)揮作用。如圖 6-4

圖 6-4

  • BeanFactory,已經(jīng)存在的 Bean 工廠接口用于獲取 Bean 對(duì)象,這次新增加了按照類型獲取 Bean 的方法: T getBean(String name, Class requiredType)
  • ListableBeanFactory,是一個(gè)擴(kuò)展 Bean 工廠接口的接口,新增加了 getBeansOfType、getBeanDefinitionNames() 方法,在 Spring 源碼中還有其他擴(kuò)展方法。
  • HierarchicalBeanFactory,在 Spring 源碼中它提供了可以獲取父類 BeanFactory 方法,屬于是一種擴(kuò)展工廠的層次子接口。Sub-interface implemented by bean factories that can be part of a hierarchy.
  • AutowireCapableBeanFactory,是一個(gè)自動(dòng)化處理Bean工廠配置的接口,目前案例工程中還沒(méi)有做相應(yīng)的實(shí)現(xiàn),后續(xù)逐步完善。
  • ConfigurableBeanFactory,可獲取 BeanPostProcessor、BeanClassLoader等的一個(gè)配置化接口。
  • ConfigurableListableBeanFactory,提供分析和修改Bean以及預(yù)先實(shí)例化的操作接口,不過(guò)目前只有一個(gè) getBeanDefinition 方法。

2. 資源加載接口定義和實(shí)現(xiàn)

  1. public interface Resource { 
  2.  
  3.     InputStream getInputStream() throws IOException; 
  4.  

在 Spring 框架下創(chuàng)建 core.io 核心包,在這個(gè)包中主要用于處理資源加載流。

定義 Resource 接口,提供獲取 InputStream 流的方法,接下來(lái)再分別實(shí)現(xiàn)三種不同的流文件操作:classPath、FileSystem、URL

ClassPath:cn.bugstack.springframework.core.io.ClassPathResource

  1. public class ClassPathResource implements Resource { 
  2.  
  3.     private final String path; 
  4.  
  5.     private ClassLoader classLoader; 
  6.  
  7.     public ClassPathResource(String path) { 
  8.         this(path, (ClassLoader) null); 
  9.     } 
  10.  
  11.     public ClassPathResource(String path, ClassLoader classLoader) { 
  12.         Assert.notNull(path, "Path must not be null"); 
  13.         this.path = path; 
  14.         this.classLoader = (classLoader != null ? classLoader : ClassUtils.getDefaultClassLoader()); 
  15.     } 
  16.  
  17.     @Override 
  18.     public InputStream getInputStream() throws IOException { 
  19.         InputStream is = classLoader.getResourceAsStream(path); 
  20.         if (is == null) { 
  21.             throw new FileNotFoundException( 
  22.                     this.path + " cannot be opened because it does not exist"); 
  23.         } 
  24.         return is
  25.     } 
  • 這一部分的實(shí)現(xiàn)是用于通過(guò) ClassLoader 讀取ClassPath 下的文件信息,具體的讀取過(guò)程主要是:classLoader.getResourceAsStream(path)
  1. public class FileSystemResource implements Resource { 
  2.  
  3.     private final File file; 
  4.  
  5.     private final String path; 
  6.  
  7.     public FileSystemResource(File file) { 
  8.         this.file = file; 
  9.         this.path = file.getPath(); 
  10.     } 
  11.  
  12.     public FileSystemResource(String path) { 
  13.         this.file = new File(path); 
  14.         this.path = path; 
  15.     } 
  16.  
  17.     @Override 
  18.     public InputStream getInputStream() throws IOException { 
  19.         return new FileInputStream(this.file); 
  20.     } 
  21.  
  22.     public final String getPath() { 
  23.         return this.path; 
  24.     } 
  25.  
  • 通過(guò)指定文件路徑的方式讀取文件信息,這部分大家肯定還是非常熟悉的,經(jīng)常會(huì)讀取一些txt、excel文件輸出到控制臺(tái)。
  1. public class UrlResource implements Resource{ 
  2.  
  3.     private final URL url; 
  4.  
  5.     public UrlResource(URL url) { 
  6.         Assert.notNull(url,"URL must not be null"); 
  7.         this.url = url; 
  8.     } 
  9.  
  10.     @Override 
  11.     public InputStream getInputStream() throws IOException { 
  12.         URLConnection con = this.url.openConnection(); 
  13.         try { 
  14.             return con.getInputStream(); 
  15.         } 
  16.         catch (IOException ex){ 
  17.             if (con instanceof HttpURLConnection){ 
  18.                 ((HttpURLConnection) con).disconnect(); 
  19.             } 
  20.             throw ex; 
  21.         } 
  22.     } 
  23.  

通過(guò) HTTP 的方式讀取云服務(wù)的文件,我們也可以把配置文件放到 GitHub 或者 Gitee 上。

3. 包裝資源加載器

按照資源加載的不同方式,資源加載器可以把這些方式集中到統(tǒng)一的類服務(wù)下進(jìn)行處理,外部用戶只需要傳遞資源地址即可,簡(jiǎn)化使用。

定義接口:cn.bugstack.springframework.core.io.ResourceLoader

  1. public interface ResourceLoader { 
  2.  
  3.     /** 
  4.      * Pseudo URL prefix for loading from the class path: "classpath:" 
  5.      */ 
  6.     String CLASSPATH_URL_PREFIX = "classpath:"
  7.  
  8.     Resource getResource(String location); 
  9.  
  • 定義獲取資源接口,里面?zhèn)鬟f location 地址即可。

實(shí)現(xiàn)接口:cn.bugstack.springframework.core.io.DefaultResourceLoader

  1. public class DefaultResourceLoader implements ResourceLoader { 
  2.  
  3.     @Override 
  4.     public Resource getResource(String location) { 
  5.         Assert.notNull(location, "Location must not be null"); 
  6.         if (location.startsWith(CLASSPATH_URL_PREFIX)) { 
  7.             return new ClassPathResource(location.substring(CLASSPATH_URL_PREFIX.length())); 
  8.         } 
  9.         else { 
  10.             try { 
  11.                 URL url = new URL(location); 
  12.                 return new UrlResource(url); 
  13.             } catch (MalformedURLException e) { 
  14.                 return new FileSystemResource(location); 
  15.             } 
  16.         } 
  17.     } 
  18.  
  • 在獲取資源的實(shí)現(xiàn)中,主要是把三種不同類型的資源處理方式進(jìn)行了包裝,分為:判斷是否為ClassPath、URL以及文件。
  • 雖然 DefaultResourceLoader 類實(shí)現(xiàn)的過(guò)程簡(jiǎn)單,但這也是設(shè)計(jì)模式約定的具體結(jié)果,像是這里不會(huì)讓外部調(diào)用放知道過(guò)多的細(xì)節(jié),而是僅關(guān)心具體調(diào)用結(jié)果即可。

4. Bean定義讀取接口

  1. public interface BeanDefinitionReader { 
  2.  
  3.     BeanDefinitionRegistry getRegistry(); 
  4.  
  5.     ResourceLoader getResourceLoader(); 
  6.  
  7.     void loadBeanDefinitions(Resource resource) throws BeansException; 
  8.  
  9.     void loadBeanDefinitions(Resource... resources) throws BeansException; 
  10.  
  11.     void loadBeanDefinitions(String location) throws BeansException; 
  12.  

這是一個(gè) Simple interface for bean definition readers. 其實(shí)里面無(wú)非定義了幾個(gè)方法,包括:getRegistry()、getResourceLoader(),以及三個(gè)加載Bean定義的方法。

這里需要注意 getRegistry()、getResourceLoader(),都是用于提供給后面三個(gè)方法的工具,加載和注冊(cè),這兩個(gè)方法的實(shí)現(xiàn)會(huì)包裝到抽象類中,以免污染具體的接口實(shí)現(xiàn)方法。

5. Bean定義抽象類實(shí)現(xiàn)

cn.bugstack.springframework.beans.factory.support.AbstractBeanDefinitionReader

  1. public abstract class AbstractBeanDefinitionReader implements BeanDefinitionReader { 
  2.  
  3.     private final BeanDefinitionRegistry registry; 
  4.  
  5.     private ResourceLoader resourceLoader; 
  6.  
  7.     protected AbstractBeanDefinitionReader(BeanDefinitionRegistry registry) { 
  8.         this(registry, new DefaultResourceLoader()); 
  9.     } 
  10.  
  11.     public AbstractBeanDefinitionReader(BeanDefinitionRegistry registry, ResourceLoader resourceLoader) { 
  12.         this.registry = registry; 
  13.         this.resourceLoader = resourceLoader; 
  14.     } 
  15.  
  16.     @Override 
  17.     public BeanDefinitionRegistry getRegistry() { 
  18.         return registry; 
  19.     } 
  20.  
  21.     @Override 
  22.     public ResourceLoader getResourceLoader() { 
  23.         return resourceLoader; 
  24.     } 
  25.  

抽象類把 BeanDefinitionReader 接口的前兩個(gè)方法全部實(shí)現(xiàn)完了,并提供了構(gòu)造函數(shù),讓外部的調(diào)用使用方,把Bean定義注入類,傳遞進(jìn)來(lái)。

這樣在接口 BeanDefinitionReader 的具體實(shí)現(xiàn)類中,就可以把解析后的 XML 文件中的 Bean 信息,注冊(cè)到 Spring 容器去了。以前我們是通過(guò)單元測(cè)試使用,調(diào)用 BeanDefinitionRegistry 完成Bean的注冊(cè),現(xiàn)在可以放到 XMl 中操作了

6. 解析XML處理Bean注冊(cè)

cn.bugstack.springframework.beans.factory.xml.XmlBeanDefinitionReader

  1. public class XmlBeanDefinitionReader extends AbstractBeanDefinitionReader { 
  2.  
  3.     public XmlBeanDefinitionReader(BeanDefinitionRegistry registry) { 
  4.         super(registry); 
  5.     } 
  6.  
  7.     public XmlBeanDefinitionReader(BeanDefinitionRegistry registry, ResourceLoader resourceLoader) { 
  8.         super(registry, resourceLoader); 
  9.     } 
  10.  
  11.     @Override 
  12.     public void loadBeanDefinitions(Resource resource) throws BeansException { 
  13.         try { 
  14.             try (InputStream inputStream = resource.getInputStream()) { 
  15.                 doLoadBeanDefinitions(inputStream); 
  16.             } 
  17.         } catch (IOException | ClassNotFoundException e) { 
  18.             throw new BeansException("IOException parsing XML document from " + resource, e); 
  19.         } 
  20.     } 
  21.  
  22.     @Override 
  23.     public void loadBeanDefinitions(Resource... resources) throws BeansException { 
  24.         for (Resource resource : resources) { 
  25.             loadBeanDefinitions(resource); 
  26.         } 
  27.     } 
  28.  
  29.     @Override 
  30.     public void loadBeanDefinitions(String location) throws BeansException { 
  31.         ResourceLoader resourceLoader = getResourceLoader(); 
  32.         Resource resource = resourceLoader.getResource(location); 
  33.         loadBeanDefinitions(resource); 
  34.     } 
  35.  
  36.     protected void doLoadBeanDefinitions(InputStream inputStream) throws ClassNotFoundException { 
  37.         Document doc = XmlUtil.readXML(inputStream); 
  38.         Element root = doc.getDocumentElement(); 
  39.         NodeList childNodes = root.getChildNodes(); 
  40.  
  41.         for (int i = 0; i < childNodes.getLength(); i++) { 
  42.             // 判斷元素 
  43.             if (!(childNodes.item(i) instanceof Element)) continue
  44.             // 判斷對(duì)象 
  45.             if (!"bean".equals(childNodes.item(i).getNodeName())) continue
  46.              
  47.             // 解析標(biāo)簽 
  48.             Element bean = (Element) childNodes.item(i); 
  49.             String id = bean.getAttribute("id"); 
  50.             String name = bean.getAttribute("name"); 
  51.             String className = bean.getAttribute("class"); 
  52.             // 獲取 Class,方便獲取類中的名稱 
  53.             Class<?> clazz = Class.forName(className); 
  54.             // 優(yōu)先級(jí) id > name 
  55.             String beanName = StrUtil.isNotEmpty(id) ? id : name
  56.             if (StrUtil.isEmpty(beanName)) { 
  57.                 beanName = StrUtil.lowerFirst(clazz.getSimpleName()); 
  58.             } 
  59.  
  60.             // 定義Bean 
  61.             BeanDefinition beanDefinition = new BeanDefinition(clazz); 
  62.             // 讀取屬性并填充 
  63.             for (int j = 0; j < bean.getChildNodes().getLength(); j++) { 
  64.                 if (!(bean.getChildNodes().item(j) instanceof Element)) continue
  65.                 if (!"property".equals(bean.getChildNodes().item(j).getNodeName())) continue
  66.                 // 解析標(biāo)簽:property 
  67.                 Element property = (Element) bean.getChildNodes().item(j); 
  68.                 String attrName = property.getAttribute("name"); 
  69.                 String attrValue = property.getAttribute("value"); 
  70.                 String attrRef = property.getAttribute("ref"); 
  71.                 // 獲取屬性值:引入對(duì)象、值對(duì)象 
  72.                 Object value = StrUtil.isNotEmpty(attrRef) ? new BeanReference(attrRef) : attrValue; 
  73.                 // 創(chuàng)建屬性信息 
  74.                 PropertyValue propertyValue = new PropertyValue(attrName, value); 
  75.                 beanDefinition.getPropertyValues().addPropertyValue(propertyValue); 
  76.             } 
  77.             if (getRegistry().containsBeanDefinition(beanName)) { 
  78.                 throw new BeansException("Duplicate beanName[" + beanName + "] is not allowed"); 
  79.             } 
  80.             // 注冊(cè) BeanDefinition 
  81.             getRegistry().registerBeanDefinition(beanName, beanDefinition); 
  82.         } 
  83.     } 
  84.  

XmlBeanDefinitionReader 類最核心的內(nèi)容就是對(duì) XML 文件的解析,把我們本來(lái)在代碼中的操作放到了通過(guò)解析 XML 自動(dòng)注冊(cè)的方式。

  • loadBeanDefinitions 方法,處理資源加載,這里新增加了一個(gè)內(nèi)部方法:doLoadBeanDefinitions,它主要負(fù)責(zé)解析 xml
  • 在 doLoadBeanDefinitions 方法中,主要是對(duì)xml的讀取 XmlUtil.readXML(inputStream) 和元素 Element 解析。在解析的過(guò)程中通過(guò)循環(huán)操作,以此獲取 Bean 配置以及配置中的 id、name、class、value、ref 信息。
  • 最終把讀取出來(lái)的配置信息,創(chuàng)建成 BeanDefinition 以及 PropertyValue,最終把完整的 Bean 定義內(nèi)容注冊(cè)到 Bean 容器:getRegistry().registerBeanDefinition(beanName, beanDefinition)

五、測(cè)試

1. 事先準(zhǔn)備

cn.bugstack.springframework.test.bean.UserDao

  1. public class UserDao { 
  2.  
  3.     private static Map<String, String> hashMap = new HashMap<>(); 
  4.  
  5.     static { 
  6.         hashMap.put("10001""小傅哥"); 
  7.         hashMap.put("10002""八杯水"); 
  8.         hashMap.put("10003""阿毛"); 
  9.     } 
  10.  
  11.     public String queryUserName(String uId) { 
  12.         return hashMap.get(uId); 
  13.     } 
  14.  

cn.bugstack.springframework.test.bean.UserService

  1. public class UserService { 
  2.  
  3.     private String uId; 
  4.  
  5.     private UserDao userDao; 
  6.  
  7.     public void queryUserInfo() { 
  8.         return userDao.queryUserName(uId); 
  9.     } 
  10.  
  11.     // ...get/set 
  • Dao、Service,是我們平常開(kāi)發(fā)經(jīng)常使用的場(chǎng)景。在 UserService 中注入 UserDao,這樣就能體現(xiàn)出Bean屬性的依賴了。

2. 配置文件

important.properties

  1. # Config File 
  2. system.key=OLpj9823dZ 

spring.xml

  1. <?xml version="1.0" encoding="UTF-8"?> 
  2. <beans> 
  3.  
  4.     <bean id="userDao" class="cn.bugstack.springframework.test.bean.UserDao"/> 
  5.  
  6.     <bean id="userService" class="cn.bugstack.springframework.test.bean.UserService"
  7.         <property name="uId" value="10001"/> 
  8.         <property name="userDao" ref="userDao"/> 
  9.     </bean> 
  10.  
  11. </beans> 

 

 

這里有兩份配置文件,一份用于測(cè)試資源加載器,另外 spring.xml 用于測(cè)試整體的 Bean 注冊(cè)功能。

3. 單元測(cè)試(資源加載)

案例

  1. private DefaultResourceLoader resourceLoader;       
  2.  
  3. @Before 
  4. public void init() { 
  5.     resourceLoader = new DefaultResourceLoader(); 
  6. }    
  7.  
  8. @Test 
  9. public void test_classpath() throws IOException { 
  10.     Resource resource = resourceLoader.getResource("classpath:important.properties"); 
  11.     InputStream inputStream = resource.getInputStream(); 
  12.     String content = IoUtil.readUtf8(inputStream); 
  13.     System.out.println(content); 
  14. }    
  15.  
  16. @Test 
  17. public void test_file() throws IOException { 
  18.     Resource resource = resourceLoader.getResource("src/test/resources/important.properties"); 
  19.     InputStream inputStream = resource.getInputStream(); 
  20.     String content = IoUtil.readUtf8(inputStream); 
  21.     System.out.println(content); 
  22. }     
  23.  
  24. @Test 
  25. public void test_url() throws IOException { 
  26.     Resource resource = resourceLoader.getResource("https://github.com/fuzhengwei/small-spring/important.properties" 
  27.     InputStream inputStream = resource.getInputStream(); 
  28.     String content = IoUtil.readUtf8(inputStream); 
  29.     System.out.println(content); 

測(cè)試結(jié)果

  1. # Config File 
  2. system.key=OLpj9823dZ 
  3.  
  4. Process finished with exit code 0 

這三個(gè)方法:test_classpath、test_file、test_url,分別用于測(cè)試加載 ClassPath、FileSystem、Url 文件,URL文件在Github,可能加載時(shí)會(huì)慢

4. 單元測(cè)試(配置文件注冊(cè)Bean)

案例

  1. @Test 
  2. public void test_xml() { 
  3.     // 1.初始化 BeanFactory 
  4.     DefaultListableBeanFactory beanFactory = new DefaultListableBeanFactory(); 
  5.  
  6.     // 2. 讀取配置文件&注冊(cè)Bean 
  7.     XmlBeanDefinitionReader reader = new XmlBeanDefinitionReader(beanFactory); 
  8.     reader.loadBeanDefinitions("classpath:spring.xml"); 
  9.  
  10.     // 3. 獲取Bean對(duì)象調(diào)用方法 
  11.     UserService userService = beanFactory.getBean("userService", UserService.class); 
  12.     String result = userService.queryUserInfo(); 
  13.     System.out.println("測(cè)試結(jié)果:" + result); 

測(cè)試結(jié)果

  1. 測(cè)試結(jié)果:小傅哥 
  2.  
  3. Process finished with exit code 0 

在上面的測(cè)試案例中可以看到,我們把以前通過(guò)手動(dòng)注冊(cè) Bean 以及配置屬性信息的內(nèi)容,交給了 new XmlBeanDefinitionReader(beanFactory) 類讀取 Spring.xml 的方式來(lái)處理,并通過(guò)了測(cè)試驗(yàn)證。

六、總結(jié)

此時(shí)的工程結(jié)構(gòu)已經(jīng)越來(lái)越有 Spring 框架的味道了,以配置文件為入口解析和注冊(cè) Bean 信息,最終再通過(guò) Bean 工廠獲取 Bean 以及做相應(yīng)的調(diào)用操作。

關(guān)于案例中每一個(gè)步驟的實(shí)現(xiàn)小傅哥這里都會(huì)盡可能參照 Spring 源碼的接口定義、抽象類實(shí)現(xiàn)、名稱規(guī)范、代碼結(jié)構(gòu)等,做相應(yīng)的簡(jiǎn)化處理。這樣大家在學(xué)習(xí)的過(guò)程中也可以通過(guò)類名或者接口和整個(gè)結(jié)構(gòu)體學(xué)習(xí) Spring 源碼,這樣學(xué)習(xí)起來(lái)就容易多了。

 

看完絕對(duì)不等于會(huì),你只有動(dòng)起手來(lái)從一個(gè)小小的工程框架結(jié)構(gòu),敲到現(xiàn)在以及以后不斷的變大、變多、變強(qiáng)時(shí),才能真的掌握這里面的知識(shí)。另外每一個(gè)章節(jié)的功能實(shí)現(xiàn)都會(huì)涉及到很多的代碼設(shè)計(jì)思路,要認(rèn)真去領(lǐng)悟。當(dāng)然實(shí)踐起來(lái)是最好的領(lǐng)悟方式!

 

責(zé)任編輯:武曉燕 來(lái)源: bugstack蟲洞棧
相關(guān)推薦

2022-04-11 08:25:37

XMLSQL語(yǔ)句Mybatis

2023-09-28 08:15:05

SpringBean加載

2017-11-21 14:56:59

2021-03-08 08:40:25

Spring Bean 創(chuàng)建單例對(duì)象

2024-01-23 08:47:13

BeanSpring加載方式

2010-09-28 10:03:15

DOM文檔對(duì)象模型

2024-05-29 08:19:03

2010-06-17 17:57:10

UML面向?qū)ο蠓治雠c設(shè)

2009-06-18 11:15:53

裝配beanxml配置Spring

2021-12-17 00:02:28

Webpack資源加載

2012-08-16 15:56:33

XML

2021-06-03 07:55:12

技術(shù)

2009-09-14 19:11:20

XML和Java Be

2022-02-07 07:10:32

服務(wù)注冊(cè)功能

2025-01-16 00:20:41

2021-06-26 16:14:11

虛擬機(jī)注冊(cè)鉤子

2010-06-12 16:30:51

UML設(shè)計(jì)

2022-04-06 09:10:40

映射器代理類接口

2009-12-29 17:47:36

Silverlight

2022-12-27 08:12:27

IOC容器Bean
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)