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

相关推荐
欧宸雅36 分钟前
Clojure语言的持续集成
开发语言·后端·golang
Bruce_Liuxiaowei1 小时前
基于Flask的DeepSeek~学术研究领域智能辅助系统设计与实现
后端·python·flask·deepseek
Asthenia04121 小时前
面试官问:你谈谈网络协议栈是什么?你觉得Java工程师需要了解哪些部分?
后端
穿林鸟2 小时前
Spring Boot项目信创国产化适配指南
java·spring boot·后端
褚翾澜2 小时前
Haskell语言的NoSQL
开发语言·后端·golang
伏游2 小时前
【BUG】生产环境死锁问题定位排查解决全过程
服务器·数据库·spring boot·后端·postgresql·bug
hycccccch3 小时前
Springcache+xxljob实现定时刷新缓存
java·后端·spring·缓存
你的人类朋友3 小时前
MQTT协议是用来做什么的?此协议常用的概念有哪些?
javascript·后端·node.js
于过3 小时前
Spring注解编程模型
java·后端