Java 反射——动态修改注解的值

反射动态修改注解的值

注解:

java 复制代码
@Target({ElementType.FIELD})
@Retention(RetentionPolicy.RUNTIME)
public @interface Value {
    String name();
}

pojo:

java 复制代码
public class User{
    
    @Value(name="张三")
    private String name;
}

现在通过注解动态将张三修改成李四,因为注解的值是在定义时就确定好的,想要修改就得通过反射进行修改,同时也可以使用外置的配置文件进行读取,类似于 Yaml 中的 @Value("${name.val}"),实际原理也是利用反射。

直接对注解进行修改:(仅展示字段上的注解)

java 复制代码
@Test
@SuppressWarnings("unchecked")
public void testModifyingAnnotations() throws Exception {
    Class<?> clazz = User.class;
    Field statType = clazz.getDeclaredField("name");
    
    Value annotation = statType.getAnnotation(Value.class);
    
    // 获取该注解,使用动态代理的方式实现修改注解的 name 值
    InvocationHandler handler = Proxy.getInvocationHandler(annotation);
    
    Field field = handler.getClass().getDeclaredField("memberValues");
    
    field.setAccessible(true);
    Map<String,Object> o = (Map<String, Object>) field.get(handler);
    
    System.out.println("原字段的注解值:"+o.get("name"));
    o.put("name","李四");
    System.out.println("修改后的注解值:"+o.get("name"));

}

结果:

text 复制代码
原字段的注解值:张三
修改后的注解值:李四

原理浅析:

可以看到我们需要修改注解的值,前提时咱们调用了 Proxy.getInvocationHandler(annotation),为什么需要用到 JDK 的代理呢?因为通过 Debug 发现 ExcelField annotation = statType.getAnnotation(ExcelField.class);

获取的注解对象 annotation,就是一个代理对象,所以我们可以通过代理对象去操作 annotation,通过观察注解对应的 InvocationHandlerAnnotationInvocationHandler 这个类。

注解的代理类有两个属性,这个很容易看出来:type 就是该注解的 Class 类型,而memberValues 就是咱们注解所存储的值了。

可以通过操作 memberValues 去操作这个注解中的值。

利用 annotation 代理对象操作注解的值

既然 AnnotationInvocationHandler 对象已经给出了注解值存储的属性,那么获取到这个属性即可。

首先获取到注解的代理对象管理器:

java 复制代码
InvocationHandler handler = Proxy.getInvocationHandler(annotation);

然后通过实际代理对象获取字段对象:memberValues

java 复制代码
Field field = handler.getClass().getDeclaredField("memberValues");

获取到属性对象:

java 复制代码
field.setAccessible(true);
Map<String, Object> memberValues = (Map<String, Object>) field.get(handler);

最后对 memberValues 进行操作:K => 注解的属性名, V => 该属性的值

java 复制代码
memberValues.set("name","李四");

到此注解的值修改完成。

总结

针对这个反射机制,有很大的操作空间再加上代理模式,我们可以在运行时进行更多的操作。

例如,现在需要根据用户的操作动态选择导出表的字段,面对这个需求,常规的 Excel 表操作肯定是不行的,所以我们可以用上反射机制:

  1. 用户根据实体类能导出的字段进行选择,然后通过 JSON,格式传递给后端
  2. 后端通过递归的方式查询和对比原有类的字段以及用户选择的字段,令用户没有选择的字段的注解值加上特殊的标识
  3. 在构建表方法中,针对注解值的特殊标识进行处理,不加入表中即可。
  4. 在导出后,最后做实体类字段注解的还原,以便用户下一次选择(因为修改后 Class 不会因为这次的方法结束而重置,直到程序重启)。

总之根据实际需求进行操作,但是一般情况下不推荐使用反射去直接修改 Class,因为操作不好,可能会导致程序出现错误,甚至安全问题。

相关推荐
码农派大星。10 分钟前
Spring Boot 配置文件
java·spring boot·后端
杜杜的man1 小时前
【go从零单排】go中的结构体struct和method
开发语言·后端·golang
幼儿园老大*1 小时前
走进 Go 语言基础语法
开发语言·后端·学习·golang·go
llllinuuu1 小时前
Go语言结构体、方法与接口
开发语言·后端·golang
cookies_s_s1 小时前
Golang--协程和管道
开发语言·后端·golang
为什么这亚子1 小时前
九、Go语言快速入门之map
运维·开发语言·后端·算法·云原生·golang·云计算
想进大厂的小王1 小时前
项目架构介绍以及Spring cloud、redis、mq 等组件的基本认识
redis·分布式·后端·spring cloud·微服务·架构
customer082 小时前
【开源免费】基于SpringBoot+Vue.JS医院管理系统(JAVA毕业设计)
java·vue.js·spring boot·后端·spring cloud·开源·intellij-idea
2402_857589362 小时前
SpringBoot框架:作业管理技术新解
java·spring boot·后端
一只爱打拳的程序猿2 小时前
【Spring】更加简单的将对象存入Spring中并使用
java·后端·spring