http://blog.csdn.net/yerenyuan_pku/article/details/52860046

上文我们已经学会使用@Resource注解注入属性。学是学会了,但也仅限于会使用而已,故我们要深入剖析其内部原理,才算真正明白。接下来我们就来编码剖析@Resource注解的实现原理,但这都是建立在用@Resource注解完成属性装配的案例基础上的。 
由于我们要使用dom4j工具来读取Sping的配置文件——beans.xml,所以需要将dom4j所需的jar包导入项目中。dom4j所需的jar包为:

  • dom4j-1.6.1.jar
  • jaxen-1.1-beta-6.jar

使用dom4j工具读取到诸如

<bean id="personService" class="cn.itcast.service.impl.PersonServiceBean">...
</bean>

这样的内容时,务必需要将读取到的bean的信息存到一个JavaBean对象中。于是我们可在junit.test包下创建这样一个JavaBean——BeanDefinition.Java,其代码为:

/*** 将读取到的bean的信息存到一个JavaBean对象中* @author li ayun**/
public class BeanDefinition { private String id; private String className; private List<PropertyDefinition> propertys = new ArrayList<PropertyDefinition>(); public BeanDefinition(String id, String className) { this.id = id; this.className = className; } public String getId() { return id; } public void setId(String id) { this.id = id; } public String getClassName() { return className; } public void setClassName(String className) { this.className = className; } public List<PropertyDefinition> getPropertys() { return propertys; } public void setPropertys(List<PropertyDefinition> propertys) { this.propertys = propertys; } }
  • 1

接着,我们还要在junit.test包下新建一个JavaBean——PropertyDefinition.java,该JavaBean专门用于存放<property ...>的信息,其代码如下:

/*** 该JavaBean专门用户存放<property>的信息* @author li ayun**/
public class PropertyDefinition {private String name;private String ref; private String value; public String getValue() { return value; } public void setValue(String value) { this.value = value; } public PropertyDefinition(String name, String ref, String value) { this.name = name; this.ref = ref; this.value = value; } public String getName() { return name; } public void setName(String name) { this.name = name; } public String getRef() { return ref; } public void setRef(String ref) { this.ref = ref; } }
  • 1

我们模拟Spring容器注入基本类型属性时,须将本身为字符串的值转成相应的属性类型的值,所以就要用到commons-beanutils工具,即要将commons-beanutils-1.9.2.jar包导入到项目中去。这样,项目中共须jar包有:
 
接下来,我们在junit.test包中新建一个注解类——ItcastResource.java,其代码如下:

@Retention(RetentionPolicy.RUNTIME)
@Target({ElementType.FIELD, ElementType.METHOD})
public @interface ItcastResource { public String name() default ""; }
  • 1

以上所有事情做好之后,我们就要对传智播客版的Spring容器修修改改了,即将ItcastClassPathXMLApplicationContext类的代码修改为:

/*** 传智播客版Spring容器* @author li ayun**/
public class ItcastClassPathXMLApplicationContext { private List<BeanDefinition> beanDefines = new ArrayList<BeanDefinition>(); private Map<String, Object> sigletons = new HashMap<String, Object>(); public ItcastClassPathXMLApplicationContext(String filename) { this.readXML(filename); this.instanceBeans(); this.injectObject(); this.annotationInject(); } private void annotationInject() { for (String beanName : sigletons.keySet()) { Object bean = sigletons.get(beanName); if (bean != null) { try { PropertyDescriptor[] ps = Introspector.getBeanInfo(bean.getClass()).getPropertyDescriptors(); for (PropertyDescriptor propertyDesc : ps) { Method setter = propertyDesc.getWriteMethod(); // 获取属性的setter方法 if (setter != null && setter.isAnnotationPresent(ItcastResource.class)) { // setter方法存在注解 ItcastResource resource = setter.getAnnotation(ItcastResource.class); Object value = null; if (resource.name() != null && !"".equals(resource.name())) { // 指定了注解的name属性 value = sigletons.get(resource.name()); } else { value = sigletons.get(propertyDesc.getName()); if (value == null) { for (String key : sigletons.keySet()) { // isAssignableFrom(xxx)方法判断propertyDesc.getPropertyType()获得的类型是否是xxx的接口或父类,或者是xxx本身 if (propertyDesc.getPropertyType().isAssignableFrom(sigletons.get(key).getClass())) { value = sigletons.get(key); break; } } } } setter.setAccessible(true); // 允许访问私有的setter方法 setter.invoke(bean, value); // 把引用对象注入到属性中 } } Field[] fields = bean.getClass().getDeclaredFields(); for (Field field : fields) { if (field.isAnnotationPresent(ItcastResource.class)) { ItcastResource resource = field.getAnnotation(ItcastResource.class); Object value = null; if (resource.name() != null && !"".equals(resource.name())) { // 指定了注解的name属性 value = sigletons.get(resource.name()); } else { value = sigletons.get(field.getName()); if (value == null) { for (String key : sigletons.keySet()) { // isAssignableFrom(xxx)方法判断field.getPropertyType()获得的类型是否是xxx的接口或父类,或者是xxx本身 if (field.getType().isAssignableFrom(sigletons.get(key).getClass())) { value = sigletons.get(key); break; } } } } field.setAccessible(true); // 允许访问private字段 field.set(bean, value); } } } catch (Exception e) { e.printStackTrace(); } } } } /** * 为bean对象的属性(依赖)注入值 */ private void injectObject() { for (BeanDefinition beanDefinition : beanDefines) { Object bean = sigletons.get(beanDefinition.getId()); if (bean != null) { try { PropertyDescriptor[] ps = Introspector.getBeanInfo(bean.getClass()).getPropertyDescriptors(); for (PropertyDefinition propertyDefinition : beanDefinition.getPropertys()) { for (PropertyDescriptor propertyDesc : ps) { if (propertyDefinition.getName().equals(propertyDesc.getName())) { Method setter = propertyDesc.getWriteMethod(); // 获取属性的setter方法,private if (setter != null) { // 最好判断有无setter方法,因为属性可以没有setter方法 /* Object value = sigletons.get(propertyDefinition.getRef()); setter.setAccessible(true); // 允许访问私有的setter方法 setter.invoke(bean, value); // 把引用对象注入到属性中 */ Object value = null; if (propertyDefinition.getRef() != null && !"".equals(propertyDefinition.getRef().trim()) ) { value = sigletons.get(propertyDefinition.getRef()); } else { // 注入基本类型 value = ConvertUtils.convert(propertyDefinition.getValue(), propertyDesc.getPropertyType()); // 把本身为字符串的值转成相应的属性类型的值 } setter.setAccessible(true); // 允许访问私有的setter方法 setter.invoke(bean, value); // 把引用对象注入到属性中 } break; } } } } catch (Exception e) { e.printStackTrace(); } } } } /** * 完成bean的实例化 */ private void instanceBeans() { for (BeanDefinition 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属性值 BeanDefinition beanDefine = new BeanDefinition(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"); // System.out.println(propertyName + "=" + propertyRef); String propertyValue= property.attributeValue("value"); PropertyDefinition propertyDefinition = new PropertyDefinition(propertyName, propertyRef, propertyValue); 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容器写好之后,我们就要试验一把了。首先将@ItcastResource注解使用在属性的setter方法上,如将PersonServiceBean类的代码修改为:

public class PersonServiceBean implements PersonService { private PersonDao personDao; private String name; @ItcastResource public void setPersonDao(PersonDao personDao) { this.personDao = personDao; } public PersonServiceBean() {} public PersonServiceBean(PersonDao personDao, String name) { this.personDao = personDao; this.name = name; } @Override public void save() { // System.out.println(name); personDao.add(); } }
  • 1

接着我们将单元测试类——SpringTest.java的代码修改为:

public class SpringTest {@Testpublic void instanceSpring() { ItcastClassPathXMLApplicationContext ctx = new ItcastClassPathXMLApplicationContext("beans.xml"); PersonService personService = (PersonService) ctx.getBean("personService"); personService.save(); } }

测试instanceSpring()方法,可看到Eclipse控制台打印: 
 
当然了,我们也可将@ItcastResource注解用在字段上,如将PersonServiceBean类的代码修改为:

public class PersonServiceBean implements PersonService { @ItcastResource private PersonDao personDao; private String name; public void setPersonDao(PersonDao personDao) { this.personDao = personDao; } public PersonServiceBean() {} public PersonServiceBean(PersonDao personDao, String name) { this.personDao = personDao; this.name = name; } @Override public void save() { // System.out.println(name); personDao.add(); } }

再次测试instanceSpring()方法,仍可看到Eclipse控制台打印: 
 
如要查看源码,可点击编码剖析@Resource注解的实现原理进行下载。

转载于:https://www.cnblogs.com/telwanggs/p/6913233.html

(转)编码剖析@Resource注解的实现原理相关推荐

  1. (转)编码剖析Spring装配基本属性的原理

    http://blog.csdn.net/yerenyuan_pku/article/details/52856465 上回我们已经讲到了Spring依赖注入的第一种方式,现在我们来详解第二种方式,须 ...

  2. (转)编码剖析Spring依赖注入的原理

    http://blog.csdn.net/yerenyuan_pku/article/details/52834561 Spring的依赖注入 前面我们就已经讲过所谓依赖注入就是指:在运行期,由外部容 ...

  3. (转)编码剖析Spring管理Bean的原理

    http://blog.csdn.net/yerenyuan_pku/article/details/52832434 在Spring的第一个案例中,我们已经知道了怎么将bean交给Spring容器进 ...

  4. Spring第七弹—依赖注入之注解方式注入及编码解析@Resource原理

        注入依赖对象可以采用手工装配或自动装配,在实际应用中建议使用手工装配,因为自动装配会产生未知情况,开发人员无法预见最终的装配结果. 手工装配依赖对象  手工装配依赖对象,在这种方式中又有两种编 ...

  5. 编码实现Spring 利用@Resource注解实现bean的注入,xml实现基本数据类型的注入

    首先分析. 1: 肯定要利用dom4j读取xml配置文件,将所有的bean的配置信息读取出来 2: 利用反射技术,实例化所有的bean 3: 写注解处理器, 利用注解和内省实现依赖对象的注入. 4: ...

  6. Spring框架你敢写精通,面试官就敢问@Autowired注解的实现原理

    面试官:Spring框架中的@Autowired注解可以标注在哪些地方? 小小白:@Autowired注解可以被标注在构造函数.属性.setter方法或配置方法上,用于实现依赖自动注入. 面试官:有没 ...

  7. Spring bean注入之注解注入-- @Autowired原理

    之前我们已经讲述过bean注入是什么了,也使用了xml的配置文件进行bean注入,这也是Spring的最原始的注入方式(xml注入). 本节课就讲注解注入. 主要讲解的注解有以下几个: @Autowi ...

  8. 深入剖析ASP.NET的编译原理之二:预编译(Precompilation)

    (转载)在本篇文章的第一部分:[原创]深入剖析ASP.NET的编译原理之一:动态编译(Dynamical Compilation),详细讨论了ASP.NET如何进行动态编译的,现在我们来谈谈另外一种重 ...

  9. 万字长文剖析清楚 Go 语言 defer 原理

    大纲 编译器怎么编译 defer `struct _defer` 数据结构 `struct _defer` 内存分配 执行 defer 函数链( `deferreturn`  ) defer 怎么传递 ...

最新文章

  1. ASP excel导出/导入Access数据库(代码+实例下载)
  2. optee中spinlock的实现原理详解
  3. html怎么显示直线,html怎么用鼠标画出一条直线,鼠标移动时候要能看到线条
  4. 基于matlab的卷积码实验报告,基于MATLAB的卷积码编译码设计仿真.doc
  5. 阿里云的云虚拟主机安装dede提示数据库连接失败的解决办法
  6. oracle 抽样_深入理解Oracle动态采样
  7. 树莓派进阶之路 (016) - 通过595驱动4位LED显示系统时间
  8. java 观察者模式示例_观察者设计模式示例
  9. 单例设计模式-静态内部类
  10. [置顶] Z-STACK之OSAL_Nv非易失性存储解读上
  11. mysql 按照条件计数_灵活的CASE...WHEN:SQL同时按条件计数按条件加和
  12. 顺序链表,动态数组实现
  13. GitHub 2200+星的任正非语录下架了,我们找到了一份fork版
  14. git 关于Git每次进入都需要输入用户名和密码的问题解决
  15. 为知笔记登录提示“打开数据库失败”解决方法
  16. php rm-rf,rm-rf误操作的恢复过程
  17. Python地理地图可视化:folium把百度地图中国城市中心经纬度解析出来并在地图上展示(三)
  18. 数据库查找姓李的人_数据库基本查询方法等
  19. 基于eNSP中大型校园/企业网络规划与设计_ensp综合大作业(ensp综合实验)
  20. 正交法---测试用例设计方法

热门文章

  1. (46)FPGA同步复位与异步复位(异步复位)
  2. (96)Verilog HDL:点灯设计
  3. (67)Verilog HDL模块条件例化
  4. (42)FPGA面试题时钟抖动和时钟偏移
  5. 判断给定的两个数是否是亲和数_动画演示LeetCode算法题:004-寻找两个有序数组的中位数...
  6. android update sdk --no-ui,CircleCI Android constraintLayout不起作用
  7. java图像风格迁移_多说迁移,Java开发模仿自主实现评论(一)
  8. mysql 创建表_每天10分钟带你学会MySQL(三)表的创建
  9. html 圆圈项目符号,html – 列表项下的项目符号
  10. Linux-oled096驱动硬件分析