- 浏览: 342666 次
- 性别:
- 来自: 上海
-
文章分类
最新评论
-
alafqq:
很好的一篇启蒙hashmap的文章;HASHTABLE的93行 ...
使用数组和链表实现hash表存储信息 -
小帅1127:
我擦,我还以为有什么大坑呢,这也写出来。。。
if..else if和if..if的区别 -
fncj:
转下http://www.dodoer.com
hadoop单机版搭建图文详解 -
yueshang520:
Spring注解原理的详细剖析与实现 -
fncj:
转下,谢谢http://www.whohelpme.com/b ...
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();
}
}
结果正常!能够引用得到(这不是废话么)
使用ref,ref所指向的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、在属性Bean:PorpertyDefinition中定义相应的属性,用来存放xml文件中定义的属性:
package com.wxy.bean;
/**
* 存放bean的property属性
*
* @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容器
* BeanDefinition的resource定位:readXML();
* BeanDefinition的载入和解析 :readXML();
* BeanDefinition在IoC容器中的注册 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资源路径,实现BeanDefinition的resource定位
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);
//获取bean的property列表节点
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()!
发表评论
-
apache日志信息详解
2011-11-06 21:19 6317一、访问日志的格式 Apache内建了记录服务器 ... -
使用CGLIB实现AOP功能与AOP概念解释 .
2011-08-19 10:30 1424第三方创建代理对象的框架cglib,目标类可以不实现接口 ... -
浏览器如何工作
2011-08-19 08:57 0http://taligarsiel.com/Projects ... -
编码实现用JDK中的Proxy实现springAOP功能
2011-08-18 15:04 791http://blog.csdn.net/iamtheevil ... -
Spring注解原理的详细剖析与实现
2011-08-14 23:09 84349本文主要分为三部分: ... -
编码剖析Spring依赖注入的原理
2011-08-10 20:01 1866一、注入依赖对象 基本类型对象注入: <b ... -
Spring的三种实例化Bean的方法
2011-08-10 14:03 1Spring的三种实例化Bean的方法 1、 使用 ... -
Spring管理bean的原理自定义实现
2011-08-10 10:44 62431、Spring通过BeanDefinition管理基于S ... -
spring环境搭建与测试
2011-08-10 08:40 3471Chapter1、搭建与测试spring的环境 1、 ... -
java回调机制实现
2011-08-08 09:06 2106Java的接口支持提供了一种获得回调的等价功能的 ... -
log4j的使用与详细分析
2011-08-05 13:32 2690一、什么是log4j? http://logging.a ... -
log4j使用详解
2011-08-04 23:05 2http://logging.apache.org/log4j ... -
spring入门及Demo
2011-07-29 16:01 0一、Spring是什么? Spring是一个开源框架,它 ... -
ibatis入门及环境搭建
2011-07-23 12:27 1468使用ibatis的理由: 1. 知道怎样操作10种以上 ... -
山寨Hibernate
2011-04-09 23:32 0---------------------------- ... -
java解析XML的四种方法的学习与比较
2011-03-30 20:55 7293四种XML解析方法: ... -
自定义日志模块实现
2011-03-30 09:58 1167package wxy.XXXX.Utils; impo ... -
synchronized(this)
2011-03-29 09:17 70551、当两个并发线程访问同一个对象object中的这个synch ... -
详细解析Java中抽象类和接口的区别(转)
2011-03-24 23:48 972在Java语言中, abstract cl ... -
首次配置hibernate碰到的各种问题
2011-03-15 21:18 15381、java.lang.NoClassDefFoundErro ...
相关推荐
Spring依赖注入(Dependency ...通过理解Spring依赖注入的实现原理,开发者能更好地控制和优化应用程序的结构,提高代码的可读性和可维护性。结合源码阅读,可以更深入地了解Spring框架的工作机制,提升开发能力。
Spring IOC(Inversion of Control,控制反转)是Spring框架的核心特性,它将对象的创建和管理权交给了Spring容器,从而解耦了应用代码与对象生命周期管理的关系。在这个例子中,我们将通过导入Excel数据来深入理解...
标题中的“第四章示例代码__自动装配”可能指的是一个教学资源或教程的一部分,它包含了一些实际的代码示例,用于解释和演示如何在Spring框架中实现自动装配。通常,这样的示例会包括配置文件(如XML或Java配置类)...
源代码分析有助于深入理解Spring的工作原理,提升编程技能,并且能够帮助开发者在遇到问题时进行调试和优化。 1. **Spring IoC容器**: Spring的核心是IoC容器,它负责管理对象的生命周期和依赖关系。通过XML配置...
深入研究Spring 2.5源码,可以了解其实现原理,如如何解析配置、如何进行依赖注入、如何实现AOP等,这对于提升Java开发者的技术水平和解决实际问题能力大有裨益。同时,通过对比不同版本的源码,还可以观察到Spring...
以上知识点涵盖了Spring中注解支持的基本原理和应用,通过深入源代码分析,可以帮助我们更好地理解和利用这些功能,提升开发效率和应用质量。对于想要深入理解Spring框架的人来说,对这部分源代码的分析是必不可少的...
在Spring框架中,类型(Type)自动装配是一种自动配置机制,它允许我们通过类型匹配来注入依赖。这种机制简化了代码,减少了...通过阅读和分析源码,我们可以深入理解Spring框架的内部工作原理,提升我们的开发技能。
【Spring 框架核心概念解析】 在Spring框架中,我们主要关注以下几个核心概念: 1. **依赖注入(Dependency ...通过下载并分析代码,我们可以更深入地理解Spring框架的工作原理以及如何在实际项目中应用这些概念。
本文将通过分析并实现一个简化版的Spring IOC+MVC框架,帮助读者深入理解这两个核心概念以及Spring的工作原理。 一、Spring IOC(Inversion of Control)——控制反转 控制反转,简称IOC,是Spring框架的核心特性...
通过分析这些代码,我们可以更好地理解Spring注解的使用和自动装配的工作原理。 总的来说,Spring注解极大地简化了Spring应用的配置,使得开发者可以更加专注于业务逻辑,而不是繁琐的XML配置。通过合理使用@...
### Spring框架的设计理念与设计模式分析 #### Spring框架的核心组件及其作用 Spring框架以其轻量级、可扩展性以及灵活性闻名于世,是现代Java应用开发的重要基石之一。本篇文章旨在深入探讨Spring框架的设计理念...
《Spring框架源代码解析教程》是一本专为深入理解Spring框架设计原理和技术细节而编写的电子书。在学习了Spring框架的基本用法和主要模块后,通过阅读源代码,我们可以进一步提升对Spring的理解,增强我们的开发技能...
总之,Spring的@Autowired自动装配是DI的重要组成部分,它简化了XML配置,提高了代码的可读性和可维护性。理解@Autowired的工作原理,能帮助我们更好地利用Spring框架构建高效的应用。通过阅读源码,可以更深入地...
总的来说,通过亲自动手实现Spring框架的一部分,我们可以更深刻地理解其设计理念和工作原理,这对于提升Java编程能力,特别是进行企业级应用开发,具有极大的帮助。在后续的部分中,我们将逐步深入到Spring的更多...
【Spring 框架概述】 Spring 是一个广泛使用的开源 Java 应用开发框架,它的核心目标是简化企业级应用的开发。...通过深入学习 Spring 源码,开发者可以更好地理解其工作原理,进一步优化和定制应用的实现。
这份"Spring源码分析"资料深入探讨了Spring的核心机制,帮助我们理解其背后的原理,从而更好地应用和优化Spring框架。 Spring的核心特性包括依赖注入(Dependency Injection, DI)、面向切面编程(Aspect-Oriented ...
**模拟Spring的IOC** 在Java世界中,Spring框架以其强大的依赖注入(Dependency Injection,简称DI)和...在实际开发中,了解并掌握Spring的IOC机制能够帮助我们更好地设计和组织代码,提升代码的灵活性和可维护性。
@Autowired 的工作原理是,在启动 Spring IoC 时,容器自动装载了一个 AutowiredAnnotationBeanPostProcessor 后置处理器,当容器扫描到 @Autowired、@Resource 或 @Inject 时,就会在 IoC 容器自动查找需要的 bean...
这个项目是对Spring框架核心功能的一个简化版本,通过逐行代码的分析,来帮助开发者理解Spring是如何通过代码配置,管理对象的生命周期,以及实现对象的依赖注入的。 文档接着详细描述了Spring框架的实现步骤,通过...
Spring5源码分析笔记旨在深入理解Spring的工作原理,帮助开发者提升技能,优化代码,以及更好地利用其核心特性。以下是对Spring5源码的一些关键知识点的详细解释: 1. **依赖注入(Dependency Injection,DI)**:...