0xZzzz 2019-06-21
在编写 SpringMVC Demo 的过程中,对自动装配有一种莫名的兴奋感,只要容器中有相对应的 Bean ,通过 @Autowire 注解,自动会将对应的 Bean 注入到相关的属性中,着的确是一个很强大的机制,但是在使用的过程中,也遇到了许多问题,其中有一个是,如果这个属性的类型有多个实现 Bean,那应该注入哪一个Bean呢,翻了很多资料也没有,只能够自己看源码,找答案。
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 则是基于注解和反射 自行自动装配