空闲的你!
在Spring中,我们可以通过 @Autowired注解的方式为一个方法中注入参数,那么这种方法背后到底发生了什么呢,这篇文章将讲述如何用Java的注解和反射实现一个“低配版”的依赖注入。
步骤:
- 通过 @interface的方式定义一个注解。
- 为某个希望杯被注入的方法添加这个注解。
- 编写测试代码,通过反射获取添加了注解的方法对应的Method对象,将该方法对象设置为可访问的,通过反射创建对象并调用这个方法,同时注入依赖数据。
根据上面的步骤来编程低配版的依赖注入,并将要编写的代码分三部分:
- Autowired: 声明的注解;
- Demo类:含有被依赖注入的方法setValue;
- Test类:通过反射获取被Autowired注解的方法,并进行依赖注入;
一:定义注解
Autowired
@Retention(RetentionPolicy.RUNTIME)
public @interface Autowired {
}
首先我们通过 @interface的方式定义的一个注解, 由此也可以看出注解的地位和类,接口类似,是一种同一级的关系。
@Retention是元注解,故名思义,它是用来注解(动词)注解(名词)的注解!(名词),RetentionPolicy.RUNTIME 表示会将这个注解保留到运行时,这样的话我们就能通过反射去处理注解了。
二. 为被注入的方法添加注解
下面我们为setStr方法添加一个注解
public class Demo {
private String value;
@Autowired
public void setValue (String value) {
this.value= value;
}
public String getValue () {
return value;
}
}
三. 通过反射处理注解
通过反射的方式获取并处理被注解的方法,将该方法对象设置为可访问的,通过反射创建对象并调用这个方法,同时注入依赖数据。
到这一步骤我们要做的事情:
- 调用Class.forName方法,传入某个类的路径字符串为参数,获取该类的Class对象。
- 通过调用该类Class对象的getDeclaredMethods方法,获得声明方法对应的Methods对象组成的数组。
- 遍历2中的Methods数组,通过调用Method对象的isAnnotationPresent方法判断该方法有没有加上Autowired注解,并对其中加上Autowired注解的方法做以下处理。
- 通过调用Method对象的setAccessible(true);方法将对象设置为可访问的,不这么搞下一步调用方法会出错。
- 通过Class对象的newInstance方法创建对象实例,假设其为object,则再通过method.invoke(object, “传入的数据")调用对象的方法,注入依赖数据。
- 将5中的对象实例object返回, 我们就获得了被注入了依赖数据的对象实例了。
代码如下:
Test.java
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
public class Test {
/**
* 这个方法会将一段文本注入到某个类中添加了@Autowired注解的方法中,并将实例对象返回
*/
public static Object injectStrToInstance (String ClassName,String value) throws ClassNotFoundException {
// 获取Demo的Class对象
Class demoClass = Class.forName(ClassName);
// 从Class对象中获取Demo中声明方法对应的Method对象
Method [] methods = demoClass.getDeclaredMethods();
for (Method method : methods) {
// 判断方法是否被加上了@Autowired这个注解
if(method.isAnnotationPresent(Autowired.class)) {
// 将方法设置为可调用的
method.setAccessible(true);
try {
Object object = demoClass.newInstance();
// 调用method方法,向其中注入value字符串
method.invoke(object,value);
return object;
} catch (IllegalAccessException e) {
e.printStackTrace();
} catch (InvocationTargetException e) {
e.printStackTrace();
} catch (InstantiationException e) {
e.printStackTrace();
}
}
}
return null;
}
public static void main (String args []) throws ClassNotFoundException {
// 进行依赖注入,并取得注入后的Demo的对象实例
Demo demo = (Demo)injectStrToInstance("Demo", "依赖注入你好!");
// 输出一下看看我们的文本是不是被成功注入进去了
System.out.println(demo.getStr());
}
}
结果如下:
依赖注入你好!
欧啦!!!