1. 程式人生 > >模擬spring建立物件

模擬spring建立物件

           java提供了反射機制,我們可以根據一個類名,創建出一個該類的一個例項。利用反射的機制,我們就很容易想出spring是怎麼樣幫我們建立物件的。

     spring的xml檔案中儲存了需要建立的bean的資訊。

     <beans>

     <bean id="persondao" class="com.shizhan.dao.impl.PersonDaoImpl">

     </bean>

     ..................

     </beans>

     其實要建立bean例項的話,我們可以根據以下的步驟:

    1:讀取xml檔案,獲取beans標籤下的所有bean標籤。

    2:遍歷所有的bean標籤

    2:得到bean標籤裡的屬性 id 和class。

    3:利用Class.forName(class).newInstance(),我們就可以構造了一個bean的例項了。

    4:我們可以把建立的bean儲存到一個map<String,Object>,其中String代表了id的值,Object代表了我們建立的bean例項。之後我們就可以通過id屬性輕鬆的從map中獲取我們建立的bean例項了。

   下面是 程式碼:

    BeanDefinition類用來儲存id和class屬性

//<bean id="PersonService1" class="com.shizhan.ps.impl.PersonServiceImpl" >
//用來儲存bean標籤的id和class屬性
public class BeanDefinition {
	private String id;
	private String className;
	
	public BeanDefinition(String id, String className) {
		this.id = id;
		this.className = className;
	}
	public String getId() {
		return id;
	}
	public void setId(String id) {
		this.id = id;
	}
	public String getClassName() {
		return className;
	}
	public void setClassName(String className) {
		this.className = className;
	}
	
}
     

import java.net.URL;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.HashSet;
import java.util.List;
import java.util.Map;

import org.dom4j.Document;
import org.dom4j.Element;
import org.dom4j.XPath;
import org.dom4j.io.SAXReader;


public class MyClassPathXMLApplicationContext {
	
	//把所有的bean標籤儲存進beanDefines中,BeanDefinition中包括了id和class
	private List<BeanDefinition> beanDefines = new ArrayList<BeanDefinition>();

	//儲存我們建立了bean物件,用bean標籤的id屬性一一對應,map<id,bean物件>
	private Map<String, Object> sigletons = new HashMap<String, Object>();
	
	public MycastClassPathXMLApplicationContext(String filename){
		this.readXML(filename);
		this.instanceBeans();
	}
	/**
	 * 完成bean的例項化
	 */
	private void instanceBeans() {
		for(BeanDefinition beanDefinition : beanDefines){
			try {
				if(beanDefinition.getClassName()!=null && 
					!"".equals(beanDefinition.getClassName().trim()))
					//利用反射原理,根據class屬性(即儲存在beanDefinition.getClassName()中)
				    //建立類名對應的物件(預設建構函式)
					Object o =  Class.forName(beanDefinition.getClassName()).newInstance();
					sigletons.put(beanDefinition.getId(),o);
			} catch (Exception e) {
				e.printStackTrace();
			}
		}
		
	}
	/**
	 * 讀取xml配置檔案
	 * @param filename
	 */
	private void readXML(String filename) {
	        SAXReader saxReader = new SAXReader();   
	        Document document=null;   
	        try{
	         URL xmlpath = this.getClass().getClassLoader()
				 .getResource(filename);
	         document = saxReader.read(xmlpath);
	         Map<String,String> nsMap = new HashMap<String,String>();
	         nsMap.put("ns","http://www.springframework.org/schema/beans");//加入名稱空間
	         XPath xsub = document.createXPath("//ns:beans/ns:bean");//建立beans/bean查詢路徑
	         xsub.setNamespaceURIs(nsMap);//設定名稱空間
	         List<Element> beans = xsub.selectNodes(document);//獲取文件下所有bean節點 
	         for(Element element: beans){
	            String id = element.attributeValue("id");//獲取id屬性值
	            String clazz = element.attributeValue("class"); //獲取class屬性值        
	            BeanDefinition beanDefine = new BeanDefinition(id, clazz);
	            beanDefines.add(beanDefine);
	         }   
	        }catch(Exception e){   
	            e.printStackTrace();
	        }
	}
	/**
	 * 獲取bean例項
	 * @param beanName
	 * @return
	 */
	public Object getBean(String beanName){
		return this.sigletons.get(beanName);
	}
}