緊接上一篇文章《輕鬆瞭解Spring中的控制反轉和依賴注入》講解了SpringIOC和DI的基本概念,這篇文章咱們模擬一下SpringIOC的工做機制,使咱們更加深入的理解其中的工做。html
類之間的結構圖以下java
如下是代碼node
BeanFactor接口:在Spring源碼中的定義是:持有對必定數量的Bean的定義,同時每一個Bean都被惟一標識的對象(類),須要實現這個接口。根據對Bean的定義,該工廠將會返回一個包含Bean定義的對象的獨立實例(原型設計模式),或者單例共享(一個不錯的單例設計模式,)範圍是整個工廠的範圍(也能夠理解成是整個容器下),返回哪一種類型的實例依賴於Bean工廠的配置:API是相同的。由於Spring2.0中擴大了依賴範圍,能夠根據具體應用上下問(如在Web環境中的請求和會話),BeanFactory是應用程序組件的中央註冊中心和集中配置。簡單的來講該接口定義了獲取Bean的方法,由子類去實現。設計模式
package ioc.factory; /** * Created by zzf on 2016/10/26. */ public interface BeanFactory { /** * 根據對象的ID標識獲取對象實例 * @param name * @return */ Object getBean(String name); }
BeanInfo類:使用Hash Map進行存儲Bean的信息,注意主要是存儲了Bean定義的Class類路徑,這樣才能經過取得type從而利用反射實例化所定義的Bean。網絡
package ioc.info; import java.lang.Object; import java.util.HashMap; import java.util.Map; /** * Created by zzf on 2016/10/26. */ public class BeanInfo { private String id; private String type; private Map<String,Object> properties=new HashMap<String,Object>(); public String getId() { return id; } public void setId(String id) { this.id = id; } public String getType() { return type; } public void setType(String type) { this.type = type; } public Map<String, Object> getProperties() { return properties; } public void setProperties(Map<String, Object> properties) { this.properties = properties; } public void addProperty(String name, Object object) { this.properties.put(name, object); } } Person:xml文件中定義的Bean package ioc.bean; /** * Created by zzf on 2016/10/26. */ public class Person { private String username; private String password; public String getUsername() { return username; } public void setUsername(String username) { this.username = username; } public String getPassword() { return password; } public void setPassword(String password) { this.password = password; } }
AbstractBeanFactory接口:是實現BeanFactory接口的抽象基類。實現獲取Bean定義的方法。是實現邏輯的最關鍵一個類,註冊、讀取、分析、注入都是在這個類上的方法調用的,具體能夠根據方法查找。是最頂層的IOC,實現該類負責從註冊器中取出註冊對象,和實現從對象描述信息轉換爲對象實例的過程實現根據名稱獲取對象的方法。dom
抽象方法setReader:由子類決定若是使用什麼樣的註冊讀取器,這裏使用了模板方法,父類定義抽象方法,但交給子類自由去設計方法內容,即交由子類本身去選擇SourceReader的實現類。ide
parseBead方法:解析生成並生成對象實例,主要經過反射完成,根據類名,加載指定類,並取得該類的Class對象使用Class對象實例化該類,獲取一個對象。逐個設置對象字段的值,這裏採用反射調用setter Method方式post
loadBeans方法:在SourceReader的實現類中所實現,主要做用是加載Xml中所定義的Bean內容,並將其屬性信息存入BeanInfo類中。this
parseBean(beanInfo):分析在loadBeans方法所存入BeanInfo,並經過反射調用注入到Person類中。spa
package ioc.factory; import ioc.info.BeanInfo; import ioc.reader.SourceReader; import java.lang.reflect.InvocationTargetException; import java.lang.reflect.Method; import java.util.Map; /** * Created by zzf on 2016/10/26. */ public abstract class AbstractBeanFactory implements BeanFactory { private String filePath; private Map<String,BeanInfo> container; protected SourceReader reader; public AbstractBeanFactory(String filePath){ this.filePath=filePath; } /** * * @param reader */ protected abstract void setReader(SourceReader reader); //註冊bean public void registerBeans(){ this.container=this.reader.loadBeans(filePath); } @Override public Object getBean(String name) { BeanInfo beanInfo=this.container.get(name); if(beanInfo==null){ return null; }else { return this.parseBean(beanInfo); } } /** * * @param beanInfo 指定對象的描述信息 * @return */ protected Object parseBean(BeanInfo beanInfo){ Class clazz; try { //加載Bean的實例 clazz=Class.forName(beanInfo.getType()); Object bean=clazz.newInstance(); //獲取該對象下的全部方法,包括私有 Method[] methods=clazz.getDeclaredMethods(); for (String property:beanInfo.getProperties().keySet()){ String setter="set"+ firstCharToUp(property); for(Method method:methods){ String methodName=method.getName(); if(methodName.equals(setter)){ Object value=beanInfo.getProperties().get(property); //使用反射調用set方法 method.invoke(bean,value); } } } return bean; } catch (ClassNotFoundException e) { e.printStackTrace(); } catch (InstantiationException e) { e.printStackTrace(); } catch (IllegalAccessException e) { e.printStackTrace(); } catch (InvocationTargetException e) { e.printStackTrace(); } return null; } private String firstCharToUp(String property) { System.out.println(property); char [] c=property.toCharArray(); String first=String.valueOf(c[0]).toUpperCase(); c[0]=first.charAt(0); System.out.println(String.valueOf(c)); return String.valueOf(c); } }
SourceReader:註冊讀取器接口,設計頂層讀取器的抽象方法,負責讀取用戶註冊的對象,繼承該接口的類能夠實現多種讀取方式,如從配置文件中讀取,根據標註讀取,從網絡中讀取等等
package ioc.reader; import ioc.info.BeanInfo; import java.util.Map; /** * Created by zzf on 2016/10/26. * */ public interface SourceReader { Map<String,BeanInfo> loadBeans(String filePath); }
XMLContext:XML的上下文,繼承了AbstractBeanFactory,裏面比較重要的方法是setReader(),在父類中該方法是抽象方法, 這樣的作的意義是交由子類實現本身所想要的讀取方式。該方法中指明註冊讀取器(這裏採用的XML,讀者能夠根據興趣去實現另外的方式如註解)並在構造該方法時一次性加載註冊的對象。
package ioc.context; import ioc.factory.AbstractBeanFactory; import ioc.reader.SourceReader; import ioc.reader.XMLSourceReader; /** * Created by zzf on 2016/10/26. * 上下文的構造方法 * */ public class XMLContext extends AbstractBeanFactory{ public XMLContext(String filePath){ super(filePath); this.setReader(new XMLSourceReader()); super.registerBeans(); } @Override protected void setReader(SourceReader reader) { super.reader=reader; } }
XmlContext類:繼承了AbstractBeanFactory抽象類,進行Bean的註冊和註冊XML的讀取器,注意返回的是Map<String, BeanInfo>
。
package ioc.reader; import ioc.info.BeanInfo; import org.dom4j.Attribute; import org.dom4j.Document; import org.dom4j.DocumentException; import org.dom4j.Element; import org.dom4j.io.SAXReader; import java.beans.IntrospectionException; import java.beans.Introspector; import java.beans.PropertyDescriptor; import java.io.InputStream; import java.io.Reader; import java.lang.reflect.InvocationTargetException; import java.lang.reflect.Method; import java.util.HashMap; import java.util.Iterator; import java.util.Map; /** * Created by zzf on 2016/10/26. * 使用 dom4j進行Xml的讀取操做 */ public class XMLSourceReader implements SourceReader { @Override public Map<String, BeanInfo> loadBeans(String filePath) { //讀取指定的配置文件 SAXReader reader = new SAXReader(); ClassLoader classLoader = Thread.currentThread().getContextClassLoader(); //從class目錄下獲取指定的xml文件 InputStream ins = classLoader.getResourceAsStream(filePath); Document doc = null; try { doc = reader.read(ins); } catch (DocumentException e) { e.printStackTrace(); } //得到根節點 Element root = doc.getRootElement(); Map<String,BeanInfo>beanInfoMap=new HashMap<String, BeanInfo>(); //遍歷bean for (Iterator i = root.elementIterator("bean"); i.hasNext();){ Element element = (Element) i.next(); //獲取bean的屬性id和class Attribute id = element.attribute("id"); Attribute cls = element.attribute("class"); try { //利用Java反射機制,經過class的名稱獲取Class對象 Class bean=Class.forName(cls.getText()); //獲取對應class的信息 java.beans.BeanInfo info= Introspector.getBeanInfo(bean); //獲取其屬性描述 PropertyDescriptor [] propertyDescriptors=info.getPropertyDescriptors(); Method method; Object object=bean.newInstance(); BeanInfo beanInfo=new BeanInfo(); for(Iterator iterator=element.elementIterator("property");iterator.hasNext();){ Element foo2= (Element) iterator.next(); //獲取該property的name屬性 Attribute name = foo2.attribute("name"); String value = null; //獲取該property的子元素value的值 for(Iterator ite1 = foo2.elementIterator("value"); ite1.hasNext();) { Element node = (Element) ite1.next(); value = node.getText(); break; } System.out.println("name:"+name.getText()+"value"+value); for (int j=0;j<propertyDescriptors.length;j++){ if(propertyDescriptors[j].getName().equalsIgnoreCase(name.getText())){ method=propertyDescriptors[j].getWriteMethod(); //利用Java的反射極致調用對象的某個set方法,並將值設置進去 method.invoke(object,value); //將獲取的對象屬性信息存入咱們自定義的BeanInfo當中 beanInfo.addProperty(name.getText(),value); } } beanInfo.setId(id.getText()); beanInfo.setType(cls.getText()); beanInfoMap.put(id.getText(),beanInfo); } return beanInfoMap; } catch (ClassNotFoundException e) { e.printStackTrace(); } catch (IntrospectionException e) { e.printStackTrace(); } catch (InstantiationException e) { e.printStackTrace(); } catch (IllegalAccessException e) { e.printStackTrace(); } catch (InvocationTargetException e) { e.printStackTrace(); } } return null; } }
<?xml version="1.0" encoding="UTF-8"?> <beans> <bean id="person" class="ioc.bean.Person"> <property name="username"> <value>zzf</value> </property> <property name="password"> <value>12345678</value> </property> </bean> </beans>
package ioc; import ioc.bean.Person; import ioc.context.XMLContext; import ioc.factory.BeanFactory; /** * Created by zzf on 2016/10/26. */ public class test { public static void main(String[] args) { BeanFactory factory=new XMLContext("configuration/config.xml"); Person person= (Person) factory.getBean("person"); System.out.println(person.getPassword()); System.out.println(person.getUsername()); } }
執行main方法後,控制檯成功輸出xml所定義的屬性。
總體的實現過程主要分如下幾步,1.讀取Xml文件,怎麼讀取?(看代碼) 2.讀取完以後呢?應該有個容器臨時存放一下,從而實現方便傳遞,這裏使用的HashMap,從中也能夠看到爲何要用BeanInfo類來存放類的信息,3.分析Map<String,BeanInfo>容器,提取裏面的Bean屬性,4.提取以後,經過反射方法注入到Person類中,從而便實現了username和password屬性的注入了。
再結合我第一篇所講的SpringIOC概念講解,相信讀者應該可以清晰的認識的IOC的做用和大概的實現過程。