文章出處

spring ioc是spring的核心之一,也是spring體系的基礎,那么spring ioc所依賴的底層技術是什么的?反射,以前我們開發程序的時候對象之間的相互調用需要用new來實現,現在所有的bean都是通過spring容器來管理。這樣做有什么好處呢?解耦!以前程序直接的調用用new直接給寫死了,現在我們可以通過注入不同的接口實現類來完成對象直接的調用。

 

首先來聊聊Java的反射機制

1、反射機制的作用:

  反編譯:.class-->.java

  通過反射機制訪問java對象的屬性,方法,構造方法等;

2、Java反射機制用途:

  在運行時判斷任意一個對象所屬的類

  在運行時構造任意一個類的對象

  在運行時判斷任意一個類所具有的成員變量和方法

  在運行時調用任意一個對象的方法

3、sun為我們提供了那些反射機制中的類:

  java.lang.Class;                
  java.lang.reflect.Constructor;           
  java.lang.reflect.Field;        
  java.lang.reflect.Method;
  java.lang.reflect.Modifier;

4、反射實現的方式

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是如果使用反射來完成對象的注入呢?

1、讀取配置文件,或者掃描注解屬性

2、根據配置文件,通過反射實例化對象

3、給對象注入依賴的屬性

4、放到類似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的思想最核心的地方在于,資源不由使用資源的雙方管理,而由不使用資源的第三方管理,這可以帶來很多好處。第一,資源集中管理,實現資源的可配置和易管理。第二,降低了使用資源雙方的依賴程度,也就是我們說的耦合度。

 


文章列表


不含病毒。www.avast.com
arrow
arrow
    全站熱搜
    創作者介紹
    創作者 AutoPoster 的頭像
    AutoPoster

    互聯網 - 大數據

    AutoPoster 發表在 痞客邦 留言(0) 人氣()