在编写 SpringMVC Demo 的过程中,对自动装配有一种莫名的兴奋感,只要容器中有相对应的 Bean ,通过 @Autowire 注解,自动会将对应的 Bean 注入到相关的属性中,着的确是一个很强大的机制,但是在使用的过程中,也遇到了许多问题,其中有一个是,如果这个属性的类型有多个实现 Bean,那应该注入哪一个Bean呢,翻了很多资料也没有,只能够自己看源码,找答案。
Java 对自动装配的支持
Java 对自动装配的支持
JDK1.5之前还没有注解功能,在1.5版本才正式加入了对注解机制的支持。
注解相当于一种标记,允许程序在运行时动态地对拥有该标记的成员进行操作。
百度百科里对注解功能的解释是:
编写文档:通过代码里标识的元数据生成文档【生成文档doc文档】
代码分析:通过代码里标识的元数据对代码进行分析【使用反射】
编译检查:通过代码里标识的元数据让编译器能够实现基本的编译检查【Override】
因此 Spring基于 JDK1.5,实现了注解功能。
那么现在我们通过自行编码,简单实现 Spring 自动装配的功能:
自定义注解:
/**
* 自定义注解,用来配置方法
*/
@Retention(RetentionPolicy.RUNTIME) // 表示注解在运行时依然存在
@Target(ElementType.METHOD) // 表示注解可以被使用于方法上
public @interface SayHiAnnotation {
String paramValue() default "johness"; // 表示我的注解需要一个参数名为"paramValue" 默认值为"johness"
自动装配:
/**
* 要使用SayHiAnnotation的元素所在类
* 由于我们定义了只有方法才能使用我们的注解,我们就使用多个方法来进行测试
*/
public class SayHiEmlement {
// 普通的方法
public void SayHiDefault(String name){
System.out.println("Hi, " + name);
}
// 使用注解并传入参数的方法
@SayHiAnnotation(paramValue="Jack")
public void SayHiAnnotation(String name){
System.out.println("Hi, " + name);
}
// 使用注解并使用默认参数的方法
@SayHiAnnotation
public void SayHiAnnotationDefault(String name){
System.out.println("Hi, " + name);
}
}
测试代码:
public class AnnotionOperator {
public static void main(String[] args) throws IllegalAccessException, IllegalArgumentException, InvocationTargetException, ClassNotFoundException {
SayHiEmlement element = new SayHiEmlement(); // 初始化一个实例,用于方法调用
Method[] methods = SayHiEmlement.class.getDeclaredMethods(); // 获得所有方法
for (Method method : methods) {
SayHiAnnotation annotationTmp = null;
if((annotationTmp = method.getAnnotation(SayHiAnnotation.class))!=null) // 检测是否使用了我们的注解
method.invoke(element,annotationTmp.paramValue()); // 如果使用了我们的注解,我们就把注解里的"paramValue"参数值作为方法参数来调用方法
else
method.invoke(element, "Rose"); // 如果没有使用我们的注解,我们就需要使用普通的方式来调用方法了
}
}
测试结果:
SayHiAnnotation()//Hi, Jack
SayHiAnnotationDefault() //Hi, johness
SayHiDefault()//Hi, Rose
可以看出,Spring 则是基于注解和反射 自行自动装配
**粗体** _斜体_ [链接](http://example.com) `代码` - 列表 > 引用
。你还可以使用@
来通知其他用户。