1.IOC原理模拟

Spring两大核心功能,IOC(Inverse  of  Control)  和 AOP(Aspect-Oriented-Programming)

IOC原理模拟:

  有这样一个beans.xml:

1 <beans>
2     <bean id="u" class="com.bjsxt.dao.impl.UserDAOImpl" />
3     <bean id="userService" class="com.bjsxt.service.UserService" >
4         <property name="userDAO" ref="u"/>
5     </bean>
6     
7 </beans>

现在我们想读取这个配置文件,实例化里面所有的bean,将其放到我们的容器中

1. 首先我们的容器是一个 Map,里面存放我们bean的实力对象

private Map<String,Object> beans = new HashMap<String,Object>();

2.开始解析这个配置文件

  a)读取这个配置文件,先获取这个配置文件的根节点,再获取根节点下所有标签名为 <bean> 的节点

1 Document doc = sb.build(this.getClass().getClassLoader().getResourceAsStream("beans.xml")); // 构造文档对象
2         Element root = doc.getRootElement();            //获取根元素
3         List list = root.getChildren("bean");           //获得根节点下所有标签名为 bean 的节点

 

  b)遍历所有的<bean>节点,拿到里面的 id 属性和 class属性

for(int i=0;i<list.size();i++) {
            Element element=(Element)list.get(i);
            String id=element.getAttributeValue("id");
            String clazz=element.getAttributeValue("class");

  

  c)根据每个bean中的class属性的值,创建这个类的一个实例,将这个实例对象作为 value,bean中的id属性作为 key ,put进我们的容器中

Object o = Class.forName(clazz).newInstance();
beans.put(id, o);      //key 为 id 属性,value为 class 属性的实例

 

  d)遍历所有的<bean>节点,判断 <bean> 节点后面是否还有 <property>节点

   如果有的话,说明 这个bean中还需要注入其他实例对象 作为 属性 注入给它,所以我们需要继续去读这个 <property>,拿到 <property> 的 name 属性 和 ref属性

for(Element propertyElement : (List<Element>)element.getChildren("property")) {            //如果某个bean中,存在property,即 存在 set***方法,则将xml文件指定的class注入进行
               String name = propertyElement.getAttributeValue("name");   //userDAO
               String ref= propertyElement.getAttributeValue("ref");   //u

 

  e)ref关联的是 某个 bean的id,所以通过ref属性,就可以 get 出 对应的实例对象,用于注入

 Object beanObject = beans.get(ref);                 //UserDAOImpl instance

  

  f)既然某个bean 希望 有 另外一个bean 可以注入进来,那么肯定要有对应 set方法,这个时候就需要用到 name 属性

  通过name属性,得到 注入的位置,如这里 name属性的值是 userDAO,说明该bean 需要一个UserDAO 类型的属性

  通过字符串拼接得到 setUserDAO方法,

String methodName = "set" + name.substring(0, 1).toUpperCase() + name.substring(1);        //拼接成setUserDAO

  

  g)反射调用该方法,完成 bean 的注入

Method m = o.getClass().getMethod(methodName, beanObject.getClass().getInterfaces()[0]);   //反射调用 setUserDAO 方法,将userDao 注入 到 userService中
m.invoke(o, beanObject);

 

  h)外界调用

  当程序一启动,就启动我们的容器,读取配置文件,完成bean 初始化和 注入,放到我们容器中

  UserService service = (UserService) applicationContext.getBean("userService");

  getBean方法,实际上就是 调用我们 容器(Map)的 get方法,通过 bean节点的id属性,拿到对象bean的实例

posted @ 2017-08-19 12:04  白日梦想家12138  阅读(217)  评论(0编辑  收藏  举报