`
freewxy
  • 浏览: 342668 次
  • 性别: Icon_minigender_1
  • 来自: 上海
社区版块
存档分类
最新评论

Spring装配基本属性的原理分析与代码实现

阅读更多

首先,做一个配置属性的基本测试。修改beans.xml,使引用外部类变成内部bean属性:

<?xml version="1.0" encoding="UTF-8"?>

<beans xmlns="http://www.springframework.org/schema/beans"

       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"

       xsi:schemaLocation="http://www.springframework.org/schema/beans

           http://www.springframework.org/schema/beans/spring-beans-2.5.xsd">

         <bean id="peopleDao" class="com.wxy.dao.impl.PeopleDaoBean"></bean>

         <bean id="peopleService" class="com.wxy.service.impl.PeopleServiceBean">

                  <property name="peopleDao" ref=” peopleDao”/>

                 

         </bean>

</beans>

 

  测试:

public class Test {

 

    public static void main(String[] args) {

        //IOC容器实例化

        ClassPathXmlApplicationContext ac = new ClassPathXmlApplicationContext("beans.xml");

        PeopleService peopleService = (PeopleService) ac.getBean("peopleService");

        peopleService.save();

    }

 

}

    结果正常!能够引用得到(这不是废话么) 

   使用refref所指向的bean可以被多个bean引用;ref依赖对象将作为一个属性被注入。

  

 

Spring是如何实现装配基本属性的呢?下面用源码模拟实现之:

 

1 xml配置文件中增加属性:

<?xml version="1.0" encoding="UTF-8"?>

<beans xmlns="http://www.springframework.org/schema/beans"

       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"

       xsi:schemaLocation="http://www.springframework.org/schema/beans

           http://www.springframework.org/schema/beans/spring-beans-2.5.xsd">

         <!--  <bean id="peopleDao" class="com.wxy.dao.impl.PeopleDaoBean"></bean>-->

         <bean id="peopleService" class="com.wxy.service.impl.PeopleServiceBean">

                    <property name="peopleDao" ref="com.wxy.dao.impl.PeopleDaoBean"/>

                    <property name="name" value="wxy"/>

                    <property name="id" value="218"/>

         </bean>

</beans>

    

2、在属性BeanPorpertyDefinition中定义相应的属性,用来存放xml文件中定义的属性:

package com.wxy.bean;

 

/**

*   存放beanproperty属性

* 

*   @creator            xiaoyu.wang  

*   @create-time     2011-8-10   下午04:40:07  

*   @revision          $Id

*/

public class PropertyDefinition {

 

    private String name; //属性名

    private String ref;  //属性依赖对象

    private String value;

 

    /**

     * @return the name

     */

    public String getName() {

        return name;

    }

 

    /**

     * @param name the name to set

     */

    public void setName(String name) {

        this.name = name;

    }

 

    /**

     * @return the ref

     */

    public String getRef() {

        return ref;

    }

 

    /**

     * @param ref the ref to set

     */

    public void setRef(String ref) {

        this.ref = ref;

    }

 

    public PropertyDefinition(String name, String ref, String value) {

        super();

        this.name = name;

        this.ref = ref;

        this.value = value;

    }

 

    /**

     * @return the value

     */

    public String getValue() {

        return value;

    }

 

    /**

     * @param value the value to set

     */

    public void setValue(String value) {

        this.value = value;

    }

 

    /* (non-Javadoc)

     * @see java.lang.Object#toString()

     */

    @Override

    public String toString() {

        return "PropertyDefinition [name=" + name + ", ref=" + ref + ", value=" + value + "]";

    }

 

}

   

3 修改PeopelServiceBean ,添加相对应的属性:

 public class PeopleServiceBean implements PeopleService {

 

    private PeopleDao peopleDao;

    private String    name;

    private Integer   id;

 

    /**

     * @return the peopleDao

     */

    public PeopleDao getPeopleDao() {

        return peopleDao;

    }

 

    /**

     * @param peopleDao the peopleDao to set

     */

    public void setPeopleDao(PeopleDao peopleDao) {

        this.peopleDao = peopleDao;

    }

 

    /**

     * @return the name

     */

    public String getName() {

        return name;

    }

 

    /**

     * @param name the name to set

     */

    public void setName(String name) {

        this.name = name;

    }

 

    /**

     * @return the id

     */

    public Integer getId() {

        return id;

    }

 

    /**

     * @param id the id to set

     */

    public void setId(Integer id) {

        this.id = id;

    }

 

    public void save() {

 

        System.out.println("--> the method is called save()! name=" + name + ",id=" + id);

        peopleDao.add();

    }

 

}

  

4、读取属性值,修改WxyClassPathXMLApplicationContext.java, 添加注入属性功能  

package com.wxy.content;

 

import java.beans.IntrospectionException;

import java.beans.Introspector;

import java.beans.PropertyDescriptor;

import java.lang.reflect.InvocationTargetException;

import java.lang.reflect.Method;

import java.net.URL;

import java.util.ArrayList;

import java.util.HashMap;

import java.util.List;

import java.util.Map;

 

import org.apache.commons.beanutils.ConvertUtils;

import org.dom4j.Document;

import org.dom4j.Element;

import org.dom4j.XPath;

import org.dom4j.io.SAXReader;

 

import com.wxy.bean.BeanDefinition;

import com.wxy.bean.PropertyDefinition;

 

/**

*  自定义IoC容器

*  BeanDefinitionresource定位:readXML();

*  BeanDefinition的载入和解析 :readXML();

*  BeanDefinitionIoC容器中的注册 instanceBeans();

*   @create-time     2011-8-10   上午09:19:17  

*   @revision          $Id

*/

public class WxyClassPathXMLApplicationContext {

 

    //存放BeanDefinition的列表,在beans.xml中定义的bean可能不止一个

    private final List<BeanDefinition> beanDefines = new ArrayList<BeanDefinition>();

    //将类名作为索引,将创建的Bean对象存入到Map

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

 

    public WxyClassPathXMLApplicationContext(String fileName) {

        //读取xml配置文件

        this.readXML(fileName);

        //实例化bean

        this.instanceBeans();

        //注入对象

        this.injectObject();

    }

 

    /**

     *bean对象的属性注入值

     */

    private void injectObject() {

        for (BeanDefinition beanDefinition : beanDefines) {

            //获取beanDefines中的对象

            Object bean = sigletons.get(beanDefinition.getId());

            if (bean != null) {

                //如果存在,利用反射技术将值注入

                try {

                    //获取Bean的描述属性类

                    PropertyDescriptor[] ps = Introspector.getBeanInfo(bean.getClass())

                        .getPropertyDescriptors();

                    for (PropertyDefinition propertyDefinition : beanDefinition.getProperties()) {

                        for (PropertyDescriptor properdesc : ps) {

                            //取得属性名字与propertyDefinition中的属性做比较

                            if (propertyDefinition.getName().equals(properdesc.getName())) {

                                //获得属性的setter方法

                                Method setter = properdesc.getWriteMethod();

                                if (setter != null) {

 

                                    Object value = null;

                                    if (propertyDefinition.getRef() != null

                                        && !propertyDefinition.getRef().isEmpty()) {//注入依赖对象

                                        value = sigletons.get(propertyDefinition.getRef());

                                    } else {//注入基本类型

                                        //将字符串类型装换成用户需要的类型,需要用到jar包中的commons-beanutils.jar

                                        value = ConvertUtils.convert(propertyDefinition.getValue(),

                                            properdesc.getPropertyType());

                                    }

                                    //允许访私有方法

                                    setter.setAccessible(true);

                                    //把引用对象注入到属性

                                    setter.invoke(bean, value);

                                }

                            }

                        }

                    }

                } catch (SecurityException e) {

                    // TODO Auto-generated catch block

                    e.printStackTrace();

                } catch (IllegalArgumentException e) {

                    // TODO Auto-generated catch block

                    e.printStackTrace();

                } catch (IntrospectionException e) {

                    // TODO Auto-generated catch block

                    e.printStackTrace();

                } catch (IllegalAccessException e) {

                    // TODO Auto-generated catch block

                    e.printStackTrace();

                } catch (InvocationTargetException e) {

                    // TODO Auto-generated catch block

                    e.printStackTrace();

                }

            }

        }

    }

 

    /**

     * 读取XML配置文件,获取BeanDefinition内容,存入到beanDefinition列表中

     * @param fileName xml配置文件名称

     */

 

    private void readXML(String fileName) {

        SAXReader saxReader = new SAXReader();

        Document document = null;

        try {

            //通过类加载器获取Resource资源路径,实现BeanDefinitionresource定位

            URL xmlPath = this.getClass().getClassLoader().getResource(fileName);

            //xml读入到document

            document = saxReader.read(xmlPath);

            Map<String, String> nsMap = new HashMap<String, String>();

            //加入命名空间

            nsMap.put("ns", "http://www.springframework.org/schema/beans");

            //创建beans/bean查询路径,注意:路径前要注明命名空间,便于解析

            XPath xsub = document.createXPath("//ns:beans/ns:bean");

            //设置命名空间

            xsub.setNamespaceURIs(nsMap);

            //获取文档下的所有Bean节点

            List<Element> beans = xsub.selectNodes(document);

            for (Element element : beans) {

                //获取id属性值

                String id = element.attributeValue("id");

                //获取class属性值

                String clazz = element.attributeValue("class");

                BeanDefinition beanDefinition = new BeanDefinition(id, clazz);

                //创建bean/property查询路径

                XPath propertysub = element.createXPath("ns:property");

                //设置命名空间

                propertysub.setNamespaceURIs(nsMap);

                //获取beanproperty列表节点

                List<Element> properties = propertysub.selectNodes(element);

                for (Element property : properties) {

                    String propertyName = property.attributeValue("name");

                    String propertyref = property.attributeValue("ref");

                    String propertyValue = property.attributeValue("value");

                    PropertyDefinition propertyDefinition = new PropertyDefinition(propertyName,

                        propertyref, propertyValue);

                    System.out.println(property);

                    //property属性添加到beanDefinition

                    beanDefinition.getProperties().add(propertyDefinition);

                }

                //将新创建的BeanDefinition赌侠ing放入到BeanDeifnitions

                beanDefines.add(beanDefinition);

            }

        } catch (Exception e) {

            System.out.println(e.toString());

        }

    }

 

    /**

     * 实例化bean,存入到sigletons

     */

    private void instanceBeans() {

        for (BeanDefinition beanDefinition : beanDefines) {

            try {

                if (beanDefinition.getClassName() != null

                    && !(beanDefinition.getClassName().isEmpty())) {

                    //利用java反射机制,生成BeanDefinition实例,并将其注册到sigletons

                    sigletons.put(beanDefinition.getId(), Class.forName(

                        beanDefinition.getClassName()).newInstance());

                }

            } catch (Exception e) {

                e.printStackTrace();

            }

        }

 

    }

 

    /**

     * 根据ID名获取实例bean

     * return 返回一个Object对象,用户使用时,需要对获取的结果进行转换类型

     */

    public Object getBean(String beanName) {

        return this.sigletons.get(beanName);

    }

}

  

5 进行测试:

public class MyTest {

 

    public static void main(String[] args) {

        //MyIOC容器实例化

        WxyClassPathXMLApplicationContext ac = new WxyClassPathXMLApplicationContext("beans.xml");

        PeopleService peopleService = (PeopleService) ac.getBean("peopleService");

        peopleService.save();

    }

 

}

  

6 测试结果:

--> the method is called save()! name=wxy,id=218

this is the method PeopleDaoBean.add()!

  

 

 

 

分享到:
评论

相关推荐

    spring依赖注入的实现原理

    Spring依赖注入(Dependency ...通过理解Spring依赖注入的实现原理,开发者能更好地控制和优化应用程序的结构,提高代码的可读性和可维护性。结合源码阅读,可以更深入地了解Spring框架的工作机制,提升开发能力。

    SpringIOC原理实现

    Spring IOC(Inversion of Control,控制反转)是Spring框架的核心特性,它将对象的创建和管理权交给了Spring容器,从而解耦了应用代码与对象生命周期管理的关系。在这个例子中,我们将通过导入Excel数据来深入理解...

    第四章示例代码__自动装配

    标题中的“第四章示例代码__自动装配”可能指的是一个教学资源或教程的一部分,它包含了一些实际的代码示例,用于解释和演示如何在Spring框架中实现自动装配。通常,这样的示例会包括配置文件(如XML或Java配置类)...

    spring源代码

    源代码分析有助于深入理解Spring的工作原理,提升编程技能,并且能够帮助开发者在遇到问题时进行调试和优化。 1. **Spring IoC容器**: Spring的核心是IoC容器,它负责管理对象的生命周期和依赖关系。通过XML配置...

    spring原代码下载

    深入研究Spring 2.5源码,可以了解其实现原理,如如何解析配置、如何进行依赖注入、如何实现AOP等,这对于提升Java开发者的技术水平和解决实际问题能力大有裨益。同时,通过对比不同版本的源码,还可以观察到Spring...

    spring源代码分析:annotation支持的实现

    以上知识点涵盖了Spring中注解支持的基本原理和应用,通过深入源代码分析,可以帮助我们更好地理解和利用这些功能,提升开发效率和应用质量。对于想要深入理解Spring框架的人来说,对这部分源代码的分析是必不可少的...

    Spring由类型(Type)自动装配源码

    在Spring框架中,类型(Type)自动装配是一种自动配置机制,它允许我们通过类型匹配来注入依赖。这种机制简化了代码,减少了...通过阅读和分析源码,我们可以深入理解Spring框架的内部工作原理,提升我们的开发技能。

    spring 代码1

    【Spring 框架核心概念解析】 在Spring框架中,我们主要关注以下几个核心概念: 1. **依赖注入(Dependency ...通过下载并分析代码,我们可以更深入地理解Spring框架的工作原理以及如何在实际项目中应用这些概念。

    spring ioc+mvc代码

    本文将通过分析并实现一个简化版的Spring IOC+MVC框架,帮助读者深入理解这两个核心概念以及Spring的工作原理。 一、Spring IOC(Inversion of Control)——控制反转 控制反转,简称IOC,是Spring框架的核心特性...

    spring注解 -----最简单的注解与自动装配例子

    通过分析这些代码,我们可以更好地理解Spring注解的使用和自动装配的工作原理。 总的来说,Spring注解极大地简化了Spring应用的配置,使得开发者可以更加专注于业务逻辑,而不是繁琐的XML配置。通过合理使用@...

    Spring框架的设计理念与设计模式分析

    ### Spring框架的设计理念与设计模式分析 #### Spring框架的核心组件及其作用 Spring框架以其轻量级、可扩展性以及灵活性闻名于世,是现代Java应用开发的重要基石之一。本篇文章旨在深入探讨Spring框架的设计理念...

    Spring框架源代码解析教程

    《Spring框架源代码解析教程》是一本专为深入理解Spring框架设计原理和技术细节而编写的电子书。在学习了Spring框架的基本用法和主要模块后,通过阅读源代码,我们可以进一步提升对Spring的理解,增强我们的开发技能...

    Spring的Autowired自动装配(XML版本+Annotation版本+源码+解析)

    总之,Spring的@Autowired自动装配是DI的重要组成部分,它简化了XML配置,提高了代码的可读性和可维护性。理解@Autowired的工作原理,能帮助我们更好地利用Spring框架构建高效的应用。通过阅读源码,可以更深入地...

    一步步实现Spring框架(一)项目搭建

    总的来说,通过亲自动手实现Spring框架的一部分,我们可以更深刻地理解其设计理念和工作原理,这对于提升Java编程能力,特别是进行企业级应用开发,具有极大的帮助。在后续的部分中,我们将逐步深入到Spring的更多...

    咕泡学院_Tom_JavaVIP课程_深入分析Spring源码(第一阶段)1

    【Spring 框架概述】 Spring 是一个广泛使用的开源 Java 应用开发框架,它的核心目标是简化企业级应用的开发。...通过深入学习 Spring 源码,开发者可以更好地理解其工作原理,进一步优化和定制应用的实现。

    Spring源码分析.rar

    这份"Spring源码分析"资料深入探讨了Spring的核心机制,帮助我们理解其背后的原理,从而更好地应用和优化Spring框架。 Spring的核心特性包括依赖注入(Dependency Injection, DI)、面向切面编程(Aspect-Oriented ...

    模拟Spring的IOC

    **模拟Spring的IOC** 在Java世界中,Spring框架以其强大的依赖注入(Dependency Injection,简称DI)和...在实际开发中,了解并掌握Spring的IOC机制能够帮助我们更好地设计和组织代码,提升代码的灵活性和可维护性。

    关于spring boot中几种注入方法的一些个人看法

    @Autowired 的工作原理是,在启动 Spring IoC 时,容器自动装载了一个 AutowiredAnnotationBeanPostProcessor 后置处理器,当容器扫描到 @Autowired、@Resource 或 @Inject 时,就会在 IoC 容器自动查找需要的 bean...

    1000行代码读懂spring核心.pdf

    这个项目是对Spring框架核心功能的一个简化版本,通过逐行代码的分析,来帮助开发者理解Spring是如何通过代码配置,管理对象的生命周期,以及实现对象的依赖注入的。 文档接着详细描述了Spring框架的实现步骤,通过...

    spring5源码分析笔记

    Spring5源码分析笔记旨在深入理解Spring的工作原理,帮助开发者提升技能,优化代码,以及更好地利用其核心特性。以下是对Spring5源码的一些关键知识点的详细解释: 1. **依赖注入(Dependency Injection,DI)**:...

Global site tag (gtag.js) - Google Analytics