如何用Java注解和反射实现依赖注入
作者:外婆的 发布时间:2022-11-04 11:34:10
概述
在Spring中,我们可以通过 @Autowired注解的方式为一个方法中注入参数,那么这种方法背后到底发生了什么呢,这篇文章将讲述如何用Java的注解和反射实现一个“低配版”的依赖注入。
下面是我们要做的一些事情:
通过 @interface的方式定义一个注解
为某个希望杯被注入的方法添加这个注解
编写测试代码,通过反射获取添加了注解的方法对应的Method对象,将该方法对象设置为可访问的,通过反射创建对象并调用这个方法,同时注入依赖数据
如上所述,我们分为三个步骤, 去加工出这个低配版的依赖注入,下面就来讲讲每一步的详细步骤
我们要编写的代码的结构分为三部分:
Autowired: 声明的注解
Demo类:含有被依赖注入的方法
setStrTest类:通过反射获取被Autowired注解的方法,并进行依赖注入
定义注解
Autowired
@Retention(RetentionPolicy.RUNTIME)
public @interface Autowired {
}
首先我们通过 @interface的方式定义的一个注解, 由此也可以看出注解的地位和类,接口类似,是一种同一级的关系
@Retention是元注解,故名思义,它是用来注解(动词)注解(名词)的注解!(名词),RetentionPolicy.RUNTIME 表示会将这个注解保留到运行时,这样的话我们就能通过反射去处理注解了。
为被注入的方法添加注解
下面我们为setStr方法添加一个注解
public class Demo {
private String str;
@Autowired
public void setStr (String str) {
this.str = str;
}
public String getStr () {
return str;
}
}
通过反射处理注解
通过反射的方式获取并处理被注解的方法,将该方法对象设置为可访问的,通过反射创建对象并调用这个方法,同时注入依赖数据
由于涉及到大量关于反射的API,所以对于反射机制话可以看看我以前写的这篇文章: https://www.cnblogs.com/penghuwan/p/7580145.html
在这一步骤我们要做的事情:
1.调用Class.forName方法,传入某个类的路径字符串为参数,获取该类的Class对象
2.通过调用该类Class对象的getDeclaredMethods方法,获得声明方法对应的Methods对象组成的数组
3.遍历2中的Methods数组,通过调用Method对象的isAnnotationPresent方法判断该方法有没有加上Autowired注解,并对其中加上Autowired注解的方法做以下处理
4.通过调用Method对象的setAccessible(true);方法将对象设置为可访问的,不这么搞下一步调用方法会出错
5.通过Class对象的newInstance方法创建对象实例,假设其为object,则再通过method.invoke(object, “传入的数据")调用对象的方法,注入依赖数据
6.将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 str) 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方法,向其中注入str字符串
method.invoke(object,str);
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 demo1 = (Demo)injectStrToInstance("Demo", "我是被注入的文本");
// 输出一下看看我们的文本是不是被成功注入进去了
System.out.println(demo1.getStr());
}
}
输出结果:
我是被注入的文本
到此为止, 我们就完成了这个低配版的依赖注入了
来源:https://www.cnblogs.com/penghuwan/p/9360602.html
![](https://www.aspxhome.com/images/zang.png)
![](https://www.aspxhome.com/images/jiucuo.png)
猜你喜欢
- 本文实例讲述了Java访问WebService返回XML数据的方法。分享给大家供大家参考。具体如下:import java.io.IOExc
- springboot用以进行web项目开发的便捷性,本文不再赘述,主要是想将工作中基于springboot与gradle的多模块项目的构建经
- 一、概述从3.1版本起,Spring框架就已经支持将缓存添加到现有的Spring应用中,和事务支持一样,缓存抽象允许在对代码影响最小的情况下
- 1. 首先新建一个shiroConfig shiro的配置类,代码如下:@Configurationpublic class SpringS
- 背景产品想对多次快速点击做一下优化,想要的效果就是双击不会打开多次但是从开发角度来说,我可以用kotlin的拓展方法来调整这个,但是之前的历
- 图片的复制无非有两种方法,一种是图片直接上传到服务器,另外一种转换成二进制流的base64码目前限chrome浏览器使用首先以um-edit
- 在[高并发Java 二] 多线程基础中,我们已经初步提到了基本的线程同步操作。这次要提到的是在并发包中的同步控制工具。1. 各种同步控制工具
- 对Jpa Entity关系映射中mappedBy的理解mappedBy 单向关系不需要设置该属性,双向关系必须设置,避免双方都建立外键字段数
- 多数据源的目的在于一个代码模块可调用多个数据库的数据进行某些业务操作。MyBatis-Plus开发者写了一个多数据源叫dynamic-dat
- 本文汇总了Spring的常用注解,以方便大家查询和使用,具体如下:使用注解之前要开启自动扫描功能其中base-package为需要扫描的包(
- 经过几天的折腾,终于到了学习一个重量级的查询方式上,使用@Query注解,使用注解有两种方式,一种是JPQL的SQL语言方式,一种是原生SQ
- 1.取整运算符取整从字面意思理解就是被除数到底包含几个除数,也就是能被整除多少次,那么它有哪些需要注意的地方呢?先看下面的两端代码: &nb
- 前言随着网络技术的发展、计算机应用水平广泛提高,原来系统的时效性、数据的正确性、操作的方便性上都存在不足,已影响到系统的正常使用。经过考察比
- 最近项目中使用了 MyBatis-Plus,点击看官方文档。使用一个新的框架,首先是验证框架的使用。 使用 MyBatis-Plu
- 前言人类建造迷宫已有5000年的历史。在世界的不同文化发展时期,这些奇特的建筑物始终吸引人们沿着弯弯曲曲、困难重重的小路吃力地行走,寻找真相
- 本文实例讲述了Java继承Thread类创建线程类。分享给大家供大家参考,具体如下:一 点睛通过继承Thread类创建线程并启动多线程的步骤
- Android中有两种主要方式使用Service,通过调用Context的startService方法或调用Context的bindServ
- 我们在使用SpringBoot进行测试的时候一般是需要加两个注解:@SpringBootTest目的是加载ApplicationContex
- java 并发线程个数的确定本文从控制变量的角度来谈决定线程个数的依据。模型很简单,在实际的生产环境中,情况肯定比下文要复杂的多。要充分的进
- 前言在前面的文章中其实大家也已经看到我使用过collect(Collectors.toList()) 将数据最后汇总成一个 List 集合。