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

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

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

1. 注解

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

@Override

下面创建自定义注解Greeter:

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

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

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

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

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

2. 修改注解

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

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

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

2.1 jdk7 实现

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

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

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

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

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

map.put(targetAnnotation, targetValue);

完整代码:

    private static final String ANNOTATIONS = "annotations";

    public static void alterAnnotationValueJDK7(
        Class<?> targetClass, Class<? extends Annotation> targetAnnotation, Annotation targetValue) {
        try {
            Field annotations = Class.class.getDeclaredField(ANNOTATIONS);
            annotations.setAccessible(true);

            Map<Class<? extends Annotation>, Annotation> map = (Map<Class<? extends Annotation>, Annotation>) annotations.get(targetClass);
            System.out.println(map);
            map.put(targetAnnotation, targetValue);
            System.out.println(map);
        } catch (Exception e) {
            e.printStackTrace();
        }
    }

2.2 jdk8实现

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

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

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

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

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

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

完整代码:

    private static final String ANNOTATION_METHOD = "annotationData";
    private static final String ANNOTATIONS = "annotations";

    public static void alterAnnotationValueJDK8(
        Class<?> targetClass, Class<? extends Annotation> 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<Class<? extends Annotation>, Annotation> map = (Map<Class<? extends Annotation>, Annotation>) annotations.get(annotationData);
            map.put(targetAnnotation, targetValue);
        } catch (Exception e) {
            e.printStackTrace();
        }
    }

3. 完整测试

首先定义类DynamicGreeter:

public class DynamicGreeter implements Greeter {

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

    @Override
    public Class<? extends Annotation> annotationType() {
        return DynamicGreeter.class;
    }

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

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() + " !!");
}

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

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

运行结果:

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

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

4. 总结

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

  • 3
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
Java反射(Reflection)是指在运行时动态地获取类的信息,包括类的属性、方法、构造器等,并可以使用这些信息来创建对象、调用方法、获取或修改属性等。Java反射是Java语言的一个特性,它使得我们可以在运行时通过获取类的信息来操作类。 Java注解(Annotation)是在Java程序中的代码中加入的一些元数据,用于对程序代码进行说明、标记和补充。Java注解可以用于对类、方法、参数、字段等进行标记和描述,使得程序开发和维护变得更加方便和简单。 Java反射和注解的原理分别如下: Java反射的原理:Java反射基于Java虚拟机在运行时对类的动态加载、链接、初始化等机制。在运行时Java虚拟机可以通过类的全限定名获取类的字节码文件,并通过类加载器将字节码文件加载到内存中。加载完成后,Java虚拟机会根据字节码文件创建一个Class对象,该对象包含了类的所有信息,包括类的属性、方法、构造器等。通过Class对象,Java反射可以获取类的信息,并使用这些信息来进行对象的创建、方法的调用、属性的获取或修改等操作。 Java注解的原理:Java注解是通过在代码中添加注解来标记和描述程序代码的信息。注解本身并不会对程序的运行产生任何影响,但可以通过反射机制获取注解信息,从而实现对程序的控制和调用。在编译时,Java编译器会将注解信息保存到类文件中,通过反射机制可以在运行时获取这些注解信息,从而实现对程序的控制和操作。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值