[置頂] Spring的DI依賴實現分析

DI(依賴注入)是Spring最底層的核心容器要實現的功能之一,利用DI能夠實現程序功能的控制反轉(控制反轉即程序之間之間的依賴關係再也不由程序員來負責,而是由Spring容器來負責)java

一個簡單的例子(DI例子)程序員

 

一個接口的源代碼(表示一我的說話)spring

package com.pp;
public interface SaySentence {
public void say();
}

一個類實現了上面的接口,表示要說的一句具體話函數

package com.pp;
public class Person {
	private Sentence sce;
	private String name;
	public Person(String name,Sentence sce){
		this.name=name;
		this.sce=sce;
	}
        public Person(){}
	public Sentence getSce() {
		return sce;
	}
	public void setSce(Sentence sce) {
		this.sce = sce;
	}
	public String getName() {
		return name;
	}
	public void setName(String name) {
		this.name = name;
	}
	public void say(){
		/*若是沒有DI機制,可能要說一句話就得以下
		 *Sentenct sce=new SaySentence();
		 *sce.say();
		 *或者由程序員顯示調用setSentence()方法設置sce
		 *或者由程序員經過構造函數的方式對sce顯示賦值
		 * */
		sce.say();
	}
}

Spring的配置文件(person.xml)post

<?xml version="1.0" encoding="UTF-8"?>
<beans
	xmlns="http://www.springframework.org/schema/beans"
	xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
	xmlns:p="http://www.springframework.org/schema/p"
	xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-3.0.xsd">
	<bean id="sentence" class="com.pp.Sentence"/>
	<bean id="person" class="com.pp.Person">
		<constructor-arg value="pp"/>
		<constructor-arg ref="sentence"/>
	</bean>
</beans>

測試的源代碼測試

 

package com.pp;
import org.springframework.beans.factory.BeanFactory;
import org.springframework.beans.factory.xml.XmlBeanFactory;
import org.springframework.core.io.ClassPathResource;
public class MainTest {
public static void main(String[] args) {
	BeanFactory factory=new XmlBeanFactory(new ClassPathResource("person.xml"));
	Person person=(Person)factory.getBean("person");
	person.say();
}
}


看到上面的,你應該也瞭解了什麼是IOC,什麼是DI,但是我想分析的不是這,這真的很簡單,我想分析就是這個DI是怎麼實現的:)下面代碼,只是摘錄,太多了,只貼出幾個關鍵函數,有興趣的,能夠共同分析this

 

 

	//加載的Bean定義
	//參數是一個資源文件,這個資源文件指定了編碼方式
	//返回值是配置文件中的bean定義個數
	public int loadBeanDefinitions(EncodedResource encodedResource) throws BeanDefinitionStoreException {
		//斷言要解析的XML文件配置存在,不能爲空
		Assert.notNull(encodedResource, "EncodedResource must not be null");
		//向日志系統輸出日誌系統,輸出XML bean的加載源
		if (logger.isInfoEnabled()) {
			logger.info("Loading XML bean definitions from " + encodedResource.getResource());
		}
    //獲取當前線程裏的ThreadLocal裏的變量集合
		Set<EncodedResource> currentResources = this.resourcesCurrentlyBeingLoaded.get();
		if (currentResources == null) {
		//若是爲空的狀況下,從新申請一下HashSet集合
			currentResources = new HashSet<EncodedResource>(4);
			this.resourcesCurrentlyBeingLoaded.set(currentResources);
		}
		//將encodeResource填加到當前線程的局部變量集合中
		if (!currentResources.add(encodedResource)) {
			throw new BeanDefinitionStoreException(
					"Detected cyclic loading of " + encodedResource + " - check your import definitions!");
		}
		try {
			InputStream inputStream = encodedResource.getResource().getInputStream();
			try {
				InputSource inputSource = new InputSource(inputStream);
				//若是設置了編譯方式,對輸入流進行編碼的設置
				if (encodedResource.getEncoding() != null) {
					inputSource.setEncoding(encodedResource.getEncoding());
				}
				return doLoadBeanDefinitions(inputSource, encodedResource.getResource());
			}
			finally {
				inputStream.close();
			}
		}
		catch (IOException ex) {
			throw new BeanDefinitionStoreException(
					"IOException parsing XML document from " + encodedResource.getResource(), ex);
		}
		finally {
		  //釋放內存空間
			currentResources.remove(encodedResource);
			if (currentResources.isEmpty()) {
				this.resourcesCurrentlyBeingLoaded.remove();
			}
		}
	}

上面這個函數最關鍵的部分是doLoadBeanDefinitions這個函數,看看它的源代碼編碼

 

 

//真正的從指定的XML文件中加載Bean的定義
	protected int doLoadBeanDefinitions(InputSource inputSource, Resource resource)
			throws BeanDefinitionStoreException {
		try {
		  //獲取資源的校驗模式
			int validationMode = getValidationModeForResource(resource);
			//文檔定義
			Document doc = this.documentLoader.loadDocument(
					inputSource, getEntityResolver(), this.errorHandler, validationMode, isNamespaceAware());
			return registerBeanDefinitions(doc, resource);
		}
		catch (BeanDefinitionStoreException ex) {
			throw ex;
		}
		catch (SAXParseException ex) {
			throw new XmlBeanDefinitionStoreException(resource.getDescription(),
					"Line " + ex.getLineNumber() + " in XML document from " + resource + " is invalid", ex);
		}
		catch (SAXException ex) {
			throw new XmlBeanDefinitionStoreException(resource.getDescription(),
					"XML document from " + resource + " is invalid", ex);
		}
		catch (ParserConfigurationException ex) {
			throw new BeanDefinitionStoreException(resource.getDescription(),
					"Parser configuration exception parsing XML from " + resource, ex);
		}
		catch (IOException ex) {
			throw new BeanDefinitionStoreException(resource.getDescription(),
					"IOException parsing XML document from " + resource, ex);
		}
		catch (Throwable ex) {
			throw new BeanDefinitionStoreException(resource.getDescription(),
					"Unexpected exception parsing XML document from " + resource, ex);
		}
	}

 

這個函數最關鍵部分是這個registerBeanDefinitions,跟蹤到裏面,略去源代碼,直接分析其最核心的代碼registerBeanDefinitionsspa

核心源碼以下線程

 

		public void registerBeanDefinitions(Document doc, XmlReaderContext readerContext) {
		this.readerContext = readerContext;
    //向日志文件輸出日誌信息
		logger.debug("Loading bean definitions");
		//獲取根元素
		Element root = doc.getDocumentElement();
    //如下是對Xml文件進行解析(各類解析,各類提取)
		BeanDefinitionParserDelegate delegate = createHelper(readerContext, root);
		preProcessXml(root);
		parseBeanDefinitions(root, delegate);
		postProcessXml(root);
	}

 

未完,待續

相關文章
相關標籤/搜索