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,因为操作不好,可能会导致程序出现错误,甚至安全问题。

相关推荐
qq_17448285757 小时前
springboot基于微信小程序的旧衣回收系统的设计与实现
spring boot·后端·微信小程序
锅包肉的九珍7 小时前
Scala的Array数组
开发语言·后端·scala
心仪悦悦7 小时前
Scala的Array(2)
开发语言·后端·scala
2401_882727578 小时前
BY组态-低代码web可视化组件
前端·后端·物联网·低代码·数学建模·前端框架
心仪悦悦8 小时前
Scala中的集合复习(1)
开发语言·后端·scala
代码小鑫9 小时前
A043-基于Spring Boot的秒杀系统设计与实现
java·开发语言·数据库·spring boot·后端·spring·毕业设计
真心喜欢你吖9 小时前
SpringBoot与MongoDB深度整合及应用案例
java·spring boot·后端·mongodb·spring
激流丶9 小时前
【Kafka 实战】Kafka 如何保证消息的顺序性?
java·后端·kafka
uzong10 小时前
一个 IDEA 老鸟的 DEBUG 私货之多线程调试
java·后端
飞升不如收破烂~10 小时前
Spring boot常用注解和作用
java·spring boot·后端