`
8366
  • 浏览: 813249 次
  • 性别: Icon_minigender_1
  • 来自: 西安
社区版块
存档分类
最新评论

第12讲 --编码剖析@Resource注解的实现原理

阅读更多

 

在这一讲开始之前,我们先学习一下jdk5.0 中的一个新特性: 注解

首先转载两篇文章:

 

J2SE5.0中新特性:注释 

 

设计自己的Annotation Java

 

 

如果使用了JSR-250中的注解,如@Resource/@PostConstruct/@PreDestroy,还需要下列jar文件
lib\j2ee\common-annotations.jar

 

 

下面我们继续改造我们自己实现的spring容器 (传智播客版spring容器)ItcastClassPathXMLApplicationContext.,完成类似的使用注解完成依赖对象的注入

 

 

步骤:

(1)首先创建一个自己的注解 ItcastResource ,类似上一讲的 Resource注解

 

package junit.test;

import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;

@Retention(RetentionPolicy.RUNTIME)
@Target({ElementType.FIELD, ElementType.METHOD})
public @interface ItcastResource {
	public String name() default ""; //注解的默认name值是""
}

 

 

@Retention(RetentionPolicy.RUNTIME)

RUNTIME表示在源码、编译好的.class文件中保留信息,在执行的时候会把这一些信息加载到JVM中去的

 

@Target({ElementType.FIELD, ElementType.METHOD})

 

@Target里面的ElementType是用来指定Annotation类型可以用在哪一些元素上的.说明一下:TYPE(类型), FIELD(属性), METHOD(方法), PARAMETER(参数), CONSTRUCTOR(构造函数),LOCAL_VARIABLE(局部变量), ANNOTATION_TYPE,PACKAGE(包),其中的TYPE(类型)是指可以用在Class,Interface,Enum和Annotation类型上.

 

(2)将 PersionServiceBean 中的Resource注解 换成我们自己的 ItcastResource 注解

 

 

(3) 修改仿spring容器 ,也就是传智播客容器 ItcastClassPathXMLApplicationContext

 

package junit.test;

import java.beans.Introspector;
import java.beans.PropertyDescriptor;
import java.lang.reflect.Field;
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;

/**
 * 传智传客版容器
 *
 */
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.annotationInject();
		this.injectObject();
	}
	/**
	 * 通过注解实现注入依赖对象
	 */
	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 properdesc : ps){
						Method setter = properdesc.getWriteMethod();//获取属性的setter方法
						/*如果setter方法不为空,并且上面有注解*/
						if(setter!=null && setter.isAnnotationPresent(ItcastResource.class)){
							/*得到注解*/
							ItcastResource resource = setter.getAnnotation(ItcastResource.class);
							Object value = null;
							//如果注解的name属性不为空,也就是这种情况	@ItcastResource(name="persionDao")
							if(resource.name()!=null && !"".equals(resource.name())){
								value = sigletons.get(resource.name());
							}else{
							/*如果注解的name为空,则根据属性的名字来得到bean*/
								value = sigletons.get(properdesc.getName());
								if(value==null)
								{
									/*如果的得到的bean为空,则根据key遍历容器中所有的bean,如果发现和属性类型相同的bean,就开始注入*/
									for(String key : sigletons.keySet()){
										if(properdesc.getPropertyType().isAssignableFrom(sigletons.get(key).getClass())){
											value = sigletons.get(key);
											break;
										}
									}
								}								
							}
							setter.setAccessible(true);
							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())){
								value = sigletons.get(resource.name());
							}else{
								value = sigletons.get(field.getName());
								if(value==null){
									for(String key : sigletons.keySet()){
										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 properdesc : ps){
							if(propertyDefinition.getName().equals(properdesc.getName())){
								Method setter = properdesc.getWriteMethod();//获取属性的setter方法 ,private
								if(setter!=null){
									Object value = null;
									if(propertyDefinition.getRef()!=null && !"".equals(propertyDefinition.getRef().trim())){
										value = sigletons.get(propertyDefinition.getRef());
									}else{
										value = ConvertUtils.convert(propertyDefinition.getValue(), properdesc.getPropertyType());
									}
									setter.setAccessible(true);
									setter.invoke(bean, value);//把引用对象注入到属性
								}
								break;
							}
						}
					}
				} catch (Exception e) {
				}
			}
		}
	}
	/**
	 * 完成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");
	            	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);
	}
}

 

 

 

 (4) 测试 将@ItcastResource(name="persionDao") 放在字段或者相应的setter方法上两种情况

 

ItcastClassPathXMLApplicationContext ctx = new ItcastClassPathXMLApplicationContext("beans.xml");
  PersionSevice ps=(PersionSevice)ctx.getBean("persionServiceBean");
  ps.save();


 (5)结果

INFO (PersionServiceBean.java:19) - 我被实例化了

INFO (PersionDaoBean.java:15) - 执行了PersionDaoBean中的add()方法

 

分享到:
评论
1 楼 laijavatoo 2012-05-10  
这个太难了

相关推荐

    Spring学习笔记(6)----编码剖析Spring依赖注入的原理

    本篇学习笔记将深入剖析Spring依赖注入的原理,通过源码分析帮助我们理解这一核心机制。 首先,依赖注入允许我们解耦组件之间的关系,使得各个组件可以独立地进行开发、测试和维护。在Spring中,DI主要通过两种方式...

    @Conditional注解应用.rar

    在Java Spring框架中,`@Conditional`注解是实现条件化bean注入的关键工具。这个注解允许我们在配置bean时设置一些条件,只有当这些条件满足时,对应的bean才会被Spring容器实例化并注入到应用程序中。这样的设计极...

    Spring Boot中的@ComponentScan注解:深入理解组件扫描机制

    Spring Boot是一个开源的Java框架,用于创建独立、微服务的...本文通过深入分析@ComponentScan注解的工作原理和使用场景,为读者提供了一个全面的指南,帮助他们在Spring Boot应用程序开发中有效地利用组件扫描功能。

    UTF-16汉字编码表

    #### 二、UTF-16编码原理 UTF-16将Unicode字符集中的每一个字符映射到一个16位或32位的数值上。对于基本多文种平面(Basic Multilingual Plane,BMP)内的字符,UTF-16使用16位编码单元进行编码;而对于BMP之外的...

    @ConfigurationProperties注解使用方法(源代码)

    1.2 第二阶段:注解配置 1.3 第三阶段:Java配置(java config) 二、@ConfigurationProperties使用方法 三、使用场景 3.1 作用于方法 3.1.1 使用场景(`配置读写分离`) 3.1.2 读写分离 案例实操 3.1.2.1 数据库的...

    如何使用Spring Boot的@Pointcut注解

    #### 第二部分:@Pointcut注解详解 1. **@Pointcut注解使用** - **方法级别**:@Pointcut注解通常用于定义具体的切点表达式。例如,可以通过`@Pointcut("execution(* com.example.demo.service.*.*(..))")`定义一...

    AMR-NB 与 AMR-WB 语音编码标准技术的对比研究

    ### AMR-NB与AMR-WB语音编码标准技术的深度对比 #### 一、引言与背景 自适应多速率语音编码技术是现代通信领域的重要组成部分,旨在提高语音通信的质量与效率。其中,AMR-NB(Adaptive Multi-Rate Narrow Band ...

    spring注解完整版[定义].pdf

    此外,Spring还提供了其他注解,如@Resource(基于名称的注入)和@Value(用于注入基本类型或SpEL表达式的结果)。这些注解共同构建了Spring的注解驱动IoC功能,使得开发更加便捷高效。 注解配置相比XML配置的优势...

    MPEG-2压缩编码技术原理应用

    ### MPEG-2压缩编码技术原理应用 #### 一、MPEG-2的系统 ##### 1. 系统的定义 MPEG-2系统是一种规范,它的主要功能是将视频、音频以及其他数据的基本流(Elementary Stream, ES)组合成一个或多个人适合存储或...

    有关pdfbox-1.3.1中Identity-H编码为乱码的解决方法

    记住,处理PDFBox和Identity-H编码问题可能需要深入理解PDF规范以及字符编码原理,但通过上述方法,你应该能找到解决问题的途径。如果问题依然存在,建议详细检查代码和文档,或者寻求社区和开发者论坛的帮助。

    P1-P4相位编码信号生成

    综上所述,P1-P4相位编码信号的生成与分析是一个涉及信号处理、编码理论和雷达原理的综合性课题。在MATLAB环境下,通过编写和运行代码,不仅可以理解相位编码的原理,还能直观地观察到各种编码信号的特性,为雷达...

    ejb3 第12讲 --开发EJB容器模型的WEB服务

    在本主题"ejb3 第12讲 --开发EJB容器模型的WEB服务"中,我们将探讨如何利用EJB 3.0版本来创建和整合Web服务。EJB 3.0是一个重要的里程碑,因为它显著简化了EJB的使用,引入了更多的POJO(Plain Old Java Object)...

    G711编码原理ppt

    这个文件名表明它是一个关于G711编码的PowerPoint演示文稿,版本号为v2,日期为2007年3月5日,可能包含了详细的讲解和示例,帮助读者深入理解μ-law和A-law编码的原理以及它们在网络语音通信中的应用。 总的来说,G...

    第三节-springboot源码解析-王炸篇.pdf

    Spring Boot源码解析是深入了解Spring Boot内部工作原理和机制的重要途径。通过源码分析,开发者可以更好地理解Spring Boot的自动装配、启动流程以及如何自定义启动器。Spring Boot的自动装配原理涉及到Spring Boot...

    基于C语言的GBK-Unicode文本编码转换系统的实现.pdf

    总结来说,本文通过分析并实现了基于C语言的GBK-Unicode文本编码转换系统,详细阐述了GBK和Unicode编码的原理、两者之间的映射关系以及转换过程中的字节序标记问题,提供了通过C语言操作映射表实现编码转换的方法,...

    数字电路分析与故障诊断 二-十进制编码器74LS147.pdf

    《数字电路分析与故障诊断:十进制编码器74LS147详解》 在计算机硬件领域,数字电路分析是理解系统运作的基础,而74LS147作为一款常用的十进制编码器,其功能和应用尤其关键。本文将深入探讨74LS147的工作原理、...

    短波3G-ALE信号链路层数据编码分析.pdf

    "短波3G-ALE信号链路层数据编码分析" 在本文中,我们将对短波3G-ALE信号链路层数据编码进行详细的分析。短波3G-ALE信号是美军新一代短波通信信号,与上一代2G-ALE信号相比,在通信链路层采用了多重数据编码方式,极...

    国家临床版3.0手术操作编码(ICD-9-CM3).xlsx.7z

    ICD-9-CM3,全称《国际疾病分类-临床修改 第九版 手术与医疗程序分类》,是世界卫生组织(WHO)制定的一套全球通用的手术和医疗程序编码系统。这个系统在国际医疗领域广泛应用,旨在为各种医疗操作提供统一的标准化...

    JAVA注解(Annotation).doc

    - 注解可以应用于类、方法、变量等,为编译器、构建工具或运行时系统提供额外的信息,例如Spring框架中的依赖注入就是通过注解实现的。 7. **注解的限制** - 同一类型的注解在同一域中只能注解一次。 - 注解不能...

Global site tag (gtag.js) - Google Analytics