自己动手写spring容器(3)
好久没有写博客了,今天闲下来将之前未完成的表达出来。
在之前的文章自己动手写spring容器(2)中完成了对spring的依赖注入的实现,这篇将会介绍spring基于注解的依赖注入的实现。
在一般的Java开发中,最常接触到的可能就是@Override和@SupressWarnings这两个注解了。使用@Override的时候只需要一个简单的声明即可。这种称为标记注解(marker annotation ),它的出现就代表了某种配置语义。而其它的注解是可以有自己的配置参数的。配置参数以名值对的方式出现。使用 @SupressWarnings的时候需要类似@SupressWarnings({"uncheck", "unused"})这样的语法。在括号里面的是该注解可供配置的值。由于这个注解只有一个配置参数,该参数的名称默认为value,并且可以省略。而花括号则表示是数组类型。在JPA中的@Table注解使用类似@Table(name = "Customer", schema = "APP")这样的语法。从这里可以看到名值对的用法。在使用注解时候的配置参数的值必须是编译时刻的常量。
从某种角度来说,可以把注解看成是一个XML元素,该元素可以有不同的预定义的属性。而属性的值是可以在声明该元素的时候自行指定的。在代码中使用注解,就相当于把一部分元数据从XML文件移到了代码本身之中,在一个地方管理和维护。
在一般的开发中,只需要通过阅读相关的API文档来了解每个注解的配置参数的含义,并在代码中正确使用即可。在有些情况下,可能会需要开发自己的注解。注解的定义有点类似接口。
首先通过开发工具向导(本文是eclipse)来生成一个注解类(通过New->Annotation来新建),如:
package com.juit; 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 YhdResource {
public String name() default "";//注解里面只能声明属性,不能声明方法,声明属性的方式比较特殊:
//语法格式为:数据类型 属性() default 默认值(默认值是可选的); 如:Stringvalue();
}
注解中定义见注释,详细的注解开发中一些量的含义大家可以百度去
要让注解实现对依赖对象的注入,必须为注解实现处理器。
在模拟spring行为的类中加入对注解的处理,
public YhdClassPathXmlApplicationContext(String fileName){
//1.读取spring的配置文件
this.readXml(fileName);
//2.实例化bean
this.instanceBeans();
//3.注解方式注入依赖对象
this.annotationInject();
//4.实现对依赖对象的注入功能
this.injectObject();
}
接下来完成annotationInject这个功能:
/**
* 注解方式注入
*
* Administer
* 2013-9-24 下午8:08:29
*/
private void annotationInject() {
//遍历所有的bean
for (String beanName : sigletons.keySet()) {
Object bean=sigletons.get(beanName);//获取需要注入的bean
if (bean != null) {
try {
//先对属性进行处理,即setter方法上标识有注解的
BeanInfo info = Introspector.getBeanInfo(bean.getClass());//通过类Introspector的getBeanInfo方法获取对象的BeanInfo 信息
PropertyDescriptor[] pds = info.getPropertyDescriptors();//获得 bean所有的属性描述
for (PropertyDescriptor pd : pds) {
Method setter=pd.getWriteMethod();//获取属性的setter方法
//属性存在setter方法,并且setter方法存在YhdResource注解
if (setter != null && setter.isAnnotationPresent(YhdResource.class)) {
YhdResource resource=setter.getAnnotation(YhdResource.class);//取得setter方法的注解
Object value=null;
//注解的name属性不为空
if (resource != null && resource.name() != null && !"".equals(resource.name())) {
value=sigletons.get(resource.name());//根据注解的name属性从容器中取出来
}else {//注解上没有标注name属性
value=sigletons.get(pd.getName());//根据属性的名称取集合中寻找此名称的bean
if (value == null) {
//没找到,遍历所有所有的bean,找类型相匹配的bean
for (String key : sigletons.keySet()) {
//判断类型是否匹配
if (pd.getPropertyType().isAssignableFrom(sigletons.get(key).getClass())) {
value=sigletons.get(key);//类型匹配的话就把此相同类型的
break;//找到了类型相同的bean,退出循环
}
}
}
}
setter.setAccessible(true);//保证setter方法可以访问私有
try {
setter.invoke(bean,value);//把引用对象注入到属性中
} catch (Exception e) {
e.printStackTrace();
}
}
} //再对字段进行处理,即对字段上标识有注解
Field[] fields=bean.getClass().getDeclaredFields();//取得声明的所有字段
for (Field field : fields) {
//判断字段上是否存在注解,若存在
if (field.isAnnotationPresent(YhdResource.class)) {
YhdResource resource=field.getAnnotation(YhdResource.class);//取得字段上的注解
Object value=null;
//字段上存在注解,并且字段上注解的name属性不为空
if (resource != null && resource.name() != null && !resource.name().equals("")) {
value=sigletons.get(resource.name());//依赖对象为根据此注解的name属性指定的对象
}else {
value=sigletons.get(field.getName());//根据字段的名称到容器中寻找bean
if (value == null) {
//没找到,根据字段的类型去寻找
for (String key : sigletons.keySet()) {
//判断类型是否匹配
if (field.getType().isAssignableFrom(sigletons.get(key).getClass())) {
value=sigletons.get(key);//类型匹配的话就把此相同类型的
break;//找到了类型相同的bean,退出循环
}
}
}
}
field.setAccessible(true);//设置允许访问私有字段
try {
field.set(bean, value);//将值为value的注入到bean对象上
} catch (Exception e) {
e.printStackTrace();
} }
}
} catch (IntrospectionException e) {
e.printStackTrace();
} }
}
}
方法中分两种情况处理,先对属性进行处理,即对setter方法上含有注解的,然后对字段进行处理,即对字段上含有注解的。
代码写完了,我们就要进行测试,把beans.xml中配置改为:
<bean id="personService" class="com.yangyang.service.impl.PersonServiceImpl">
</bean>
然后在业务方法的类中添加注解:
此处先测试setter方法上有注解的:
package com.yangyang.service.impl; import com.juit.YhdResource;
import com.yangyang.dao.PersonDao;
import com.yangyang.model.Person;
import com.yangyang.service.PersonService; public class PersonServiceImpl implements PersonService{ public PersonDao getPersonDao() {
return personDao;
}
@YhdResource
public void setPersonDao(PersonDao personDao) {
this.personDao = personDao;
} @Override
public void savePerson() {
System.out.println("service中的save方法调用成功");
personDao.savePerson();
} }
执行单元测试方法:
@Test
public void testInstanceSping() {
YhdClassPathXmlApplicationContext ctx=new YhdClassPathXmlApplicationContext("resources/beans.xml");
PersonService personService=(PersonService)ctx.getBean("personService");
personService.savePerson();
}
可以看到控制台打印出:service中的save方法调用成功
dao中的save方法调用成功
再测试字段上有注解的:
package com.yangyang.service.impl; import com.juit.YhdResource;
import com.yangyang.dao.PersonDao;
import com.yangyang.model.Person;
import com.yangyang.service.PersonService; public class PersonServiceImpl implements PersonService{
@YhdResource
private PersonDao personDao; public PersonDao getPersonDao() {
return personDao;
}
public void setPersonDao(PersonDao personDao) {
this.personDao = personDao;
} @Override
public void savePerson() {
System.out.println("service中的save方法调用成功");
personDao.savePerson();
} }
同样的可以看到控制台打印:
service中的save方法调用成功
dao中的save方法调用成功
这样注解来实现依赖对象的注入就基本上完成了。
自己动手写spring容器(3)的更多相关文章
- 自己动手写spring容器(2)
上篇我们自己写了一个很简单的spring容器,该容器只是做了简单的bean的实例化,并没有spring的核心之一的IOC(依赖注入),也叫做控制反转,这里我就不讲这个的具体含义,不知道的园友可以自行百 ...
- 自己动手写spring容器(1)
毕业刚刚一年多一点,毕业了后也顺利的进入了一家著名的互联网公司,做的是后台系统,用的呢也是SSI(struts2,spring)框架,平时做做项目,也已足够了,但是感觉越来越没动力了,越来越没有激情了 ...
- 自己动手写Spring框架--IOC、MVC
对于一名Java开发人员,我相信没有人不知道 Spring 框架,而且也能够轻松就说出 Spring 的特性-- IOC.MVC.AOP.ORM(batis). 下面我想简单介绍一下我写的轻量级的 S ...
- 自己动手编写spring IOC源码
前言:对于spring IOC概念不是很了解的朋友可以阅读我上一篇博客--轻松理解spring IOC(这两篇博客也是由于我的个人原因导致现在才发布,惭愧啊).通过这篇博客的理解之后,相信大家会对sp ...
- [原]容器学习(一):动手模拟spring的IoC
介绍 学习经典框架的实现原理以及设计模式在其实际中的运用,是非常有必要的,可以让我们更好进行面向对象. 本篇文章就来模拟Spring的IOC功能,明白原理后,可以更好的使用它,进而为进行面向对象提供一 ...
- 容器学习(一):动手模拟spring的IoC
介绍 学习经典框架的实现原理以及设计模式在事实上际中的运用,是很有必要的,能够让我们更好进行面向对象. 本篇文章就来模拟Spring的IOC功能.明确原理后,能够更好的使用它,进而为进行面向对象提供一 ...
- 我自横刀向天笑,手写Spring IOC容器,快来Look Look!
目录 IOC分析 IOC是什么 IOC能够带来什么好处 IOC容器是做什么工作的 IOC容器是否是工厂模式的实例 IOC设计实现 设计IOC需要什么 定义接口 一:Bean工厂接口 二:Bean定义的 ...
- 手写Spring,定义标记类型Aware接口,实现感知容器对象
作者:小傅哥 博客:https://bugstack.cn 沉淀.分享.成长,让自己和他人都能有所收获! 一.前言 同事写的代码,我竟丝毫看不懂! 大佬的代码,就像 "赖蛤蟆泡青蛙,张的丑玩 ...
- 自己动手开发IOC容器
前两天写简历.写了一句:精通Spring IoC容器.怎么个精通法?还是自己动手写个IOC容器吧. 什么是IoC(Inversion of Control)?什么是DI(Dependency Inje ...
随机推荐
- Ubuntu下的用户和权限(二)
五.chown.chgrp命令 从名字就能够猜測他们是干嘛的,可是这两个命令须要root权限. chown命令的格式为:chown user:group file 中间的user : group三项 ...
- (C语言)共用体union的使用方法举例
曾经在学校学习C语言的时候一直搞不懂那个共用体union有什么用的.工作之后才发现它的一些妙用,现举比例如以下: 1. 为了方便看懂代码. 比方说想写一个3 * 3的矩阵,能够这样写: [ 注:以下用 ...
- DDD分层架构之聚合
DDD分层架构之聚合 前面已经介绍了DDD分层架构的实体和值对象,本文将介绍聚合以及与其高度相关的并发主题. 我在之前已经说过,初学者第一步需要将业务逻辑尽量放到实体或值对象中,给实体“充血”,这样可 ...
- 【WebSocket初探
】
众所周知,socket是编写网络通信应用的基本技术,网络数据交换大多直接或间接通过socket进行.对于直接使用socket的client与服务端,一旦连接被建立则均可主动向对方传送数据,而对于使用更 ...
- SSIS中执行SQL任务组件参数传递的问题
原文:SSIS中执行SQL任务组件参数传递的问题 症状: 执行SQL任务,传递参数到子查询中,执行报错. 错误: 失败,错误如下:"无法从使用 sub-select 查询的 SQL 语句中派 ...
- HTTP 报文中的 Header 字段进行身份验证
[小技巧][ASP.Net MVC Hack] 使用 HTTP 报文中的 Header 字段进行身份验证 在一些 Web 系统中,身份验证是依靠硬件证书进行的:在电脑上插入 USB 证书,浏览器插件读 ...
- Windows 7上使用HP QC的问题
C(Quantity Center)是一款不错的测试管理工具,最近把公司的操作系统从Windows XP升级到Windows 7之后,发现登录到QC Server的Addin页面,很多客户端组件不能正 ...
- visual studio code, asp.net5, mvc6资料汇总
最近在试探性地跟随微软最新发布的一些产品,现列下某些挺好的文章和链接 code.visualstudio.com http://blogs.msdn.com/b/cesardelatorre/arch ...
- SpringMVC之 数据绑定-1
SpringMVC学习系列(4) 之 数据绑定-1 在系列(3)中我们介绍了请求是如何映射到一个action上的,下一步当然是如何获取到请求中的数据,这就引出了本篇所要讲的内容—数据绑定. 首先看一下 ...
- 封装两个简单的Jquery组件
Jquery给我们提供了很大的方便,我们把他看成是一个公共库,以致在这个公共库上延伸出了很多Jquery插件:在项目过程中,有些插件总是不那么令人满意: 主要说两个项目用途: 1. 遮罩层,跟一般的 ...