spring ioc是spring的核心之一,也是spring體系的基礎,那麼spring ioc所依賴的底層技術是什麼的?反射,之前咱們開發程序的時候對象之間的相互調用須要用new來實現,如今全部的bean都是經過spring容器來管理。這樣作有什麼好處呢?解耦!之前程序直接的調用用new直接給寫死了,如今咱們能夠經過注入不一樣的接口實現類來完成對象直接的調用。java
首先來聊聊Java的反射機制 spring
一、反射機制的做用:ide
反編譯:.class-->.java學習
經過反射機制訪問java對象的屬性,方法,構造方法等;網站
二、Java反射機制用途: this
在運行時判斷任意一個對象所屬的類 spa
在運行時構造任意一個類的對象 code
在運行時判斷任意一個類所具備的成員變量和方法 orm
在運行時調用任意一個對象的方法 xml
三、sun爲咱們提供了那些反射機制中的類:
java.lang.Class;
java.lang.reflect.Constructor;
java.lang.reflect.Field;
java.lang.reflect.Method;
java.lang.reflect.Modifier;
四、反射實現的方式
Class c=Class.forName("className"); 註明:className必須爲全名,也就是得包含包名,好比,cn.xx.UserInfo; Object obj=c.newInstance(); //建立對象的實例 Constructor getConstructor(Class[] params) //根據指定參數得到public構造器 Constructor[] getConstructors() //得到public的全部構造器 Constructor getDeclaredConstructor(Class[] params) //根據指定參數得到public和非public的構造器 Constructor[] getDeclaredConstructors() //得到public的全部構造器 ewInstance(); //建立對象的實例 得到類方法的方法 Method getMethod(String name, Class[] params), 根據方法名,參數類型得到方法 Method[] getMethods() //得到全部的public方法 Method getDeclaredMethod(String name, Class[] params) //根據方法名和參數類型,得到public和非public的方法 Method[] getDeclaredMethods() //得到因此的public和非public方法 得到類中屬性的方法 Field getField(String name) //根據變量名獲得相應的public變量 Field[] getFields() //得到類中因此public的方法 Field getDeclaredField(String name) //根據方法名得到public和非public變量 Field[] getDeclaredFields() //得到類中全部的public和非public方法
總結一句,之前寫的代碼,類的熟悉、方法什麼東西的都固定了,若是使用反射咱們就能夠在運行的時候動態的去修改、增刪對象的熟悉、方法等等。
spring IOC是若是使用反射來完成對象的注入呢?
一、讀取配置文件,或者掃描註解屬性
二、根據配置文件,經過反射實例化對象
三、給對象注入依賴的屬性
四、放到相似hashMap結構中,供系統調用
/** * 學習版容器 * */
public class LeamClassPathXMLApplicationContext { private List<Definition> beanDefines = new ArrayList<Definition>(); private Map<String, Object> sigletons = new HashMap<String, Object>(); public LeamClassPathXMLApplicationContext(String filename){ this.readXML(filename); this.instanceBeans(); this.injectObject(); } /** * 爲bean對象的屬性注入值 */ private void injectObject() { for(Definition beanDefinition : beanDefines){ Object bean = sigletons.get(beanDefinition.getId()); if(bean!=null){ try { PropertyDescriptor[] ps = Introspector.getBeanInfo(bean.getClass()).getPropertyDescriptors(); for(ProsDefinition propertyDefinition : beanDefinition.getPropertys()){ for(PropertyDescriptor properdesc : ps){ if(propertyDefinition.getName().equals(properdesc.getName())){ Method setter = properdesc.getWriteMethod();//獲取屬性的setter方法 if(setter!=null){ Object value = sigletons.get(propertyDefinition.getRef()); setter.setAccessible(true); setter.invoke(bean, value);//把引用對象注入到屬性 } break; } } } } catch (Exception e) { } } } } /** * 完成bean的實例化 */ private void instanceBeans() { for(Definition beanDefinition : beanDefines){ try { if(beanDefinition.getClassName()!=null && !"".equals(beanDefinition.getClassName().trim())) sigletons.put(beanDefinition.getId(), Class.forName(beanDefinition.getClassName()).newInstance()); } 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屬性值 Definition beanDefine = new Definition(id, clazz); XPath propertysub = element.createXPath("ns:property"); propertysub.setNamespaceURIs(nsMap);//設置命名空間 List<Element> propertys = propertysub.selectNodes(element); for(Element property : propertys){ String propertyName = property.attributeValue("name");//元素內部引用的屬性也獲取 String propertyref = property.attributeValue("ref"); ProsDefinition propertyDefinition = new ProsDefinition(propertyName, propertyref); beanDefine.getPropertys().add(propertyDefinition); } beanDefines.add(beanDefine); } }catch(Exception e){ e.printStackTrace(); }
} /** * 獲取bean實例 * @param beanName * @return */ public Object getBean(String beanName){ return this.sigletons.get(beanName); } }
spring ioc核心思想
ioc的思想最核心的地方在於,資源不禁使用資源的雙方管理,而由不使用資源的第三方管理,這能夠帶來不少好處。第一,資源集中管理,實現資源的可配置和易管理。第二,下降了使用資源雙方的依賴程度,也就是咱們說的耦合度。