ICode9

精准搜索请尝试: 精确搜索
首页 > 编程语言> 文章详细

Java 在运行时修改注解参数值

2021-10-10 13:31:28  阅读:208  来源: 互联网

标签:map Class Java class 参数值 annotations 注解 Annotation


转自 https://blog.csdn.net/neweastsun/article/details/109276023

Java 在运行时修改注解参数值

注解是在java代码中增加一种元数据,这些元数据可以嵌入在class文件中在编译时处理,也可以保留至运行时通过Reflection进行访问。本文讨论如何在运行时修改注解值,我们示例使用类级别注解。

1. 注解

Java允许使用现有注解创建新的注解。最简单的注释形式是@符号后接注释名:

@Override

   
   
  • 1

下面创建自定义注解Greeter:

@Retention(RetentionPolicy.RUNTIME)
public @interface Greeter {    
    public String greet() default ""; 
}

   
   
  • 1
  • 2
  • 3
  • 4

现在我们创建类Greeting并增加类级别注解:

@Greeter(greet="Good morning")
public class Greetings {}

   
   
  • 1
  • 2

现在可以使用反射访问注解,Java的Class提供方法getAnnotation获取类的注解:

Greeter greetings = Greetings.class.getAnnotation(Greeter.class);
System.out.println("Hello there, " + greetings.greet() + " !!");

   
   
  • 1
  • 2

2. 修改注解

Java Class类通过map管理注解:Annotation 类作为key,Annotation 对象作为值:

Map<Class<? extends Annotation>, Annotation> map;

   
   
  • 1

我们可以在运行时更新map在,在jdk7和jdk8中访问该map有差异。

2.1 jdk7 实现

Class类有annotations私有属性,为了访问该属性,需设置其可访问性为true。java提供getDeclaredField 方法通过名称访问属性。

Field annotations = Class.class.getDeclaredField(ANNOTATIONS);
annotations.setAccessible(true);

   
   
  • 1
  • 2

现在可以访问目标类注解map:

Map<Class<? extends Annotation>, Annotation> map = annotations.get(targetClass);

   
   
  • 1

该map包含所有注解及其值的信息。如果想修改Greeter注解值,可以通过更新注解对象:

map.put(targetAnnotation, targetValue);

   
   
  • 1

完整代码:

    private static final String ANNOTATIONS = "annotations";
public static void alterAnnotationValueJDK7(
    Class&lt;?&gt; targetClass, Class&lt;? extends Annotation&gt; targetAnnotation, Annotation targetValue) {
    try {
        Field annotations = Class.class.getDeclaredField(ANNOTATIONS);
        annotations.setAccessible(true);

        Map&lt;Class&lt;? extends Annotation&gt;, Annotation&gt; map = (Map&lt;Class&lt;? extends Annotation&gt;, Annotation&gt;) annotations.get(targetClass);
        System.out.println(map);
        map.put(targetAnnotation, targetValue);
        System.out.println(map);
    } catch (Exception e) {
        e.printStackTrace();
    }
}
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16

2.2 jdk8实现

java8 通过AnnotationData类存储注解信息,我们可以通过annotationData方法访问该对象。同样也需要设置该方法的可访问性为true:

Method method = Class.class.getDeclaredMethod(ANNOTATION_METHOD, null);
method.setAccessible(true);

 
 
  • 1
  • 2

现在可以访问annotations字段,同样也需要设置该其访问属性:

Field annotations = annotationData.getClass().getDeclaredField(ANNOTATIONS);
annotations.setAccessible(true);

 
 
  • 1
  • 2

通过annotations获取存储注解类和值的map,通过map可以修改注解值:

Map<Class<? extends Annotation>, Annotation> map = annotations.get(annotationData); 
map.put(targetAnnotation, targetValue);

 
 
  • 1
  • 2

完整代码:

    private static final String ANNOTATION_METHOD = "annotationData";
    private static final String ANNOTATIONS = "annotations";
public static void alterAnnotationValueJDK8(
    Class&lt;?&gt; targetClass, Class&lt;? extends Annotation&gt; targetAnnotation, Annotation targetValue) {
    try {
        Method method = Class.class.getDeclaredMethod(ANNOTATION_METHOD, null);
        method.setAccessible(true);

        Object annotationData = method.invoke(targetClass);

        Field annotations = annotationData.getClass().getDeclaredField(ANNOTATIONS);
        annotations.setAccessible(true);

        Map&lt;Class&lt;? extends Annotation&gt;, Annotation&gt; map = (Map&lt;Class&lt;? extends Annotation&gt;, Annotation&gt;) annotations.get(annotationData);
        map.put(targetAnnotation, targetValue);
    } catch (Exception e) {
        e.printStackTrace();
    }
}
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20

3. 完整测试

首先定义类DynamicGreeter:

public class DynamicGreeter implements Greeter {
private String greet;
public DynamicGreeter(String greet) {
    this.greet = greet;
}

@Override
public Class&lt;? extends Annotation&gt; annotationType() {
    return DynamicGreeter.class;
}

@Override
public String greet() {
    return greet;
}

}

  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17

main函数中进行测试:

public static void main(String ...args) {
    Greeter greetings = Greetings.class.getAnnotation(Greeter.class);
    System.err.println("Hello there, " + greetings.greet() + " !!");
Greeter targetValue = new DynamicGreeter("Good evening");
alterAnnotationValueJDK8(Greetings.class, Greeter.class, targetValue);
//alterAnnotationValueJDK7(Greetings.class, Greeter.class, targetValue);

greetings = Greetings.class.getAnnotation(Greeter.class);
System.err.println("Hello there, " + greetings.greet() + " !!");

}

  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11

初始时注解的值为“Good morning” ,接着现再创建一个值为“Good evening”的Greeter类对象。然后通过上面定义的方法修改注解值:

alterAnnotationValueJDK8(Greetings.class, Greeter.class, targetValue);

 
 
  • 1

运行结果:

Hello there, Good morning !!
Hello there, Good evening !!

 
 
  • 1
  • 2

成功在运行时修改了注解的值。

4. 总结

Java使用两个数据字段来存储注解数据:annotation、declaredAnnotations。两者之间的区别:第一个存储来自父类的注解,之后一个仅存储当前类的注解。由于JDK 7和JDK 8中实现getAnnotation有所不同,为了简单起见在这里使用annotations字段map。

标签:map,Class,Java,class,参数值,annotations,注解,Annotation
来源: https://blog.csdn.net/ZHAI_KE/article/details/120685303

本站声明: 1. iCode9 技术分享网(下文简称本站)提供的所有内容,仅供技术学习、探讨和分享;
2. 关于本站的所有留言、评论、转载及引用,纯属内容发起人的个人观点,与本站观点和立场无关;
3. 关于本站的所有言论和文字,纯属内容发起人的个人观点,与本站观点和立场无关;
4. 本站文章均是网友提供,不完全保证技术分享内容的完整性、准确性、时效性、风险性和版权归属;如您发现该文章侵犯了您的权益,可联系我们第一时间进行删除;
5. 本站为非盈利性的个人网站,所有内容不会用来进行牟利,也不会利用任何形式的广告来间接获益,纯粹是为了广大技术爱好者提供技术内容和技术思想的分享性交流网站。

专注分享技术,共同学习,共同进步。侵权联系[81616952@qq.com]

Copyright (C)ICode9.com, All Rights Reserved.

ICode9版权所有