- 浏览: 809138 次
- 性别:
- 来自: 西安
文章分类
- 全部博客 (307)
- struts (8)
- hibernate (3)
- spring (32)
- opensourceproject (12)
- javaScript (9)
- primeton EOS (2)
- journey of heart (10)
- Design pattern (6)
- ejb (17)
- point (37)
- Linux&Unix (22)
- ibatis (10)
- AJAX (6)
- DB (26)
- Protocol (6)
- chart (4)
- web server (11)
- webservice (7)
- integration (3)
- tuxedo (5)
- ext (4)
- android (1)
- c/c++ (12)
- JVM (1)
- paginationFrame (2)
- code (2)
- report (1)
- High-performance web (1)
- svn (1)
- JQuery (1)
- workDaily (2)
- cloud (16)
- Python (8)
- English (2)
- shell (5)
- googleCode (1)
- nio (1)
- hyper-v (1)
- debug (3)
- vbs (2)
- openstack (3)
- K8S (1)
- Mesos (0)
- Spark (0)
- Marathon (0)
最新评论
-
钱图大展:
chao2751021 写道lib包哪里去下载,找不到
大型网站用户行为记录的一个实现--基于clickStream(第一部分) -
钱图大展:
无法下载
大型网站用户行为记录的一个实现--基于clickStream(第一部分) -
fm395728572:
shell脚本中用到了环境变量,但是获取不到,例如脚本中有一句 ...
ganymed-ssh2 for Java -
liuhanjiang:
我qq147229234
大型网站用户行为记录的一个实现--基于clickStream(第一部分) -
liuhanjiang:
博主 我利用您提供的方法实现博文中介绍的clickstream ...
大型网站用户行为记录的一个实现--基于clickStream(第一部分)
在这一讲开始之前,我们先学习一下jdk5.0 中的一个新特性: 注解
首先转载两篇文章:
如果使用了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()方法
发表评论
-
hibernatetemplate类使用
2010-03-19 22:25 1205http://115893520.iteye.com/blog ... -
Spring 中的JdbcTemplate使用
2010-03-19 22:11 35101.JdbcTemplate的execute()方 ... -
SpringAOP 的使用(两种方式)
2009-10-22 13:42 1526使用Spring AOP 拦截 方法,给被拦截的 ... -
第30讲--Spring提供的CharacterEncoding和OpenSessionInView功能
2009-09-19 18:24 1798CharacterEncoding: 在web ... -
第29讲--为Spring集成的Hibernate配置二级缓存
2009-09-18 00:29 3404合理的使用缓存策略,往往在web开发中提高性能起到关键 ... -
第28讲--Struts与Spring集成方案2(Spring集成Struts)
2009-09-17 00:57 1647集成步骤: 1.和方案1一样需要在web. ... -
第27讲--Struts与Spring集成方案1(Struts集成Spring)
2009-09-16 08:25 1834Spring2.5+Hibernate3.3+Stru ... -
第25,26讲 --搭建和配置Spring与Hibernate整合的环境
2009-06-21 11:21 1611Spring2.5+Hibernate3.3+Stru ... -
第24讲 --使用Spring配置文件实现事务管理
2009-06-20 23:56 1773采用基于XML方式配置事务 <bean ... -
第23讲 --使用Spring注解方式管理事务与传播行为详解
2009-06-11 23:19 2252事务传播属性 REQUIR ... -
第21,22讲 --搭建和配置Spring与jdbc整合的环境+Spring集成的jdbc编码和事务
2009-06-10 08:06 2702Spring+JDBC组合开发配置数据源有两种方式: ... -
第20讲--aspectj的切入点语法定义细节
2009-06-06 17:47 3644表达式分析: expression=" ... -
第19讲 --使用Spring配置文件实现AOP
2009-06-06 17:21 1550上面我们是使用注解的方式实现spring AOP的,下面 ... -
第18讲 --使用Spring的注解方式实现AOP的细节
2009-06-06 15:48 1567上一讲我们使用spring注解的方式理由aop技术 ... -
第17讲 --使用Spring的注解方式实现AOP入门
2009-06-05 00:48 2276使用Spring进行面向切面(AOP)编程 要 ... -
第16讲 --使用CGLIB实现AOP功能与AOP概念解释
2009-06-04 07:42 1642当代理对象没有实现 ... -
第15讲 --使用JDK中的Proxy技术实现AOP功能
2009-06-04 07:34 2337当目标类实现了接口,我们可以使用jdk的Proxy ... -
第14讲 --让Spring自动扫描和管理Bean
2009-06-02 23:40 8757通过在classpath自动扫描方式把组件纳入sp ... -
第13讲 --Autowire注解与自动装配
2009-06-02 22:46 31391. Autowire 默认是 ... -
第11讲 --用@Resource注解完成属性装配
2009-05-31 22:42 7045bean的注入方式有3种: 第一种:使用构造器注 ...
相关推荐
这篇文章将深入探讨`@Resource`注解的工作原理,以及它如何与Spring的IoC(Inversion of Control,控制反转)容器协同工作。 首先,让我们理解`@Resource`的基本用法。当在类的字段或方法上使用`@Resource`时,...
在Spring框架中,`@Autowired`和`@Resource`注解是两个常见的依赖注入(DI, Dependency Injection)工具,它们都是用来解决组件之间的耦合问题,使得代码更加灵活和可测试。然而,这两个注解在具体使用时有一些关键性...
原理分析 ----- 在 Spring 中,@PathVariable 注解的参数是通过 PathVariableMethodArgumentResolver.resolveName() 方法获取的,该方法会根据 URL template 变量来解析参数。在 RequestMappingInfoHandlerMapping....
在该示例中,如果不添加 @Configuration 注解,dmzService 将被创建两次,第一次是由 Spring 容器创建的,第二次是由 a() 方法调用 dmzService() 方法时创建的。这将导致单例被破坏, Bean 的生命周期管理失效的问题...
Spring Boot是一个开源的Java框架,用于创建独立、微服务的...本文通过深入分析@ComponentScan注解的工作原理和使用场景,为读者提供了一个全面的指南,帮助他们在Spring Boot应用程序开发中有效地利用组件扫描功能。
#### 二、UTF-16编码原理 UTF-16将Unicode字符集中的每一个字符映射到一个16位或32位的数值上。对于基本多文种平面(Basic Multilingual Plane,BMP)内的字符,UTF-16使用16位编码单元进行编码;而对于BMP之外的...
1.2 第二阶段:注解配置 1.3 第三阶段:Java配置(java config) 二、@ConfigurationProperties使用方法 三、使用场景 3.1 作用于方法 3.1.1 使用场景(`配置读写分离`) 3.1.2 读写分离 案例实操 3.1.2.1 数据库的...
"卷积编码与解码的MATLAB实现及性能分析" 本资源摘要信息是关于卷积编码与解码的MATLAB实现及性能分析的课程设计报告。下面是相关的知识点: 1、卷积编码的基本概念:卷积编码是一种通用的-channel coding技术,...
在这个例子中,`Employee`类代表了一个Excel工作表,其中`name`字段对应于第一行第一列,`age`字段对应于第一行第二列。 在实际应用中,我们还可以利用注解来定义更复杂的逻辑,如数据验证、样式设置、公式计算等。...
在“LTE系统中Reed-Muller码的编码研究与实现”这篇毕业设计中,作者可能详细介绍了Reed-Muller码的理论背景,分析了其在LTE环境下的性能优势,并给出了具体的软件或硬件实现方案。可能还包含了对不同级别Reed-...
74LS148是一款常见的集成电路,它实现了一个8输入、3输出的优先编码器功能。在这个设计中,我们将深入探讨74LS148的工作原理、8-3优先编码器的功能,以及如何在Proteus软件中进行电路仿真。 74LS148是TTL系列芯片,...
此外,Spring还提供了其他注解,如@Resource(基于名称的注入)和@Value(用于注入基本类型或SpEL表达式的结果)。这些注解共同构建了Spring的注解驱动IoC功能,使得开发更加便捷高效。 注解配置相比XML配置的优势...
### MPEG-2压缩编码技术原理应用 #### 一、MPEG-2的系统 ##### 1. 系统的定义 MPEG-2系统是一种规范,它的主要功能是将视频、音频以及其他数据的基本流(Elementary Stream, ES)组合成一个或多个人适合存储或...
它基于线性预测编码(LPC)原理,通过分析语音信号的频谱特性,用较小的数据量来近似重构原始语音信号。在这个压缩包中,我们拥有C语言实现的LPC-10编码器和解码器的源代码,对于理解和研究这种编码技术非常有价值。...
这个文件名表明它是一个关于G711编码的PowerPoint演示文稿,版本号为v2,日期为2007年3月5日,可能包含了详细的讲解和示例,帮助读者深入理解μ-law和A-law编码的原理以及它们在网络语音通信中的应用。 总的来说,G...
本文将详尽解读“三菱PLC程序源码-电子开料锯程序附注解.zip”中的内容,带领读者深入理解PLC编程原理,并结合电子开料锯的具体应用,探讨实际工程中的编程技巧。 一、PLC基础与三菱PLC概述 PLC是一种专为工业环境...
本文将围绕“三菱PLC程序源码-裁切机带伺服带中文注解”这一主题,深入探讨三菱PLC如何实现裁切机的伺服控制,并结合源码分析,帮助读者理解其中的关键技术和实现方法。 1. PLC与伺服系统的配合 在裁切机中,伺服...
实现自定义注解的第一步是定义注解类。例如: ```java @Retention(RetentionPolicy.RUNTIME) @Target(ElementType.FIELD) public @interface BindView { int value(); } ``` 这里,`@Retention(RetentionPolicy....
Spring Boot源码解析是深入了解Spring Boot内部工作原理和机制的重要途径。通过源码分析,开发者可以更好地理解Spring Boot的自动装配、启动流程以及如何自定义启动器。Spring Boot的自动装配原理涉及到Spring Boot...