引言
如果没有用来读取注解的类和方法,那么注解也就不会比注释更有用处了。使用注解的过程中,很重要的一部分就是创建使用注解处理器。Java SE5扩展了反射机制的API,以帮助程序员快速的构造自定义注解处理器。
注解处理器类库
Java使用Annotation接口来代表程序元素前面的注解,该接口是所有Annotation类型的父接口。除此之外,Java在java.lang.reflect 包下新增了AnnotatedElement接口,该接口代表程序中可以接受注解的程序元素,该接口主要有如下几个实现类:
Class:类定义
Constructor:构造器定义
Field:累的成员变量定义
Method:类的方法定义
Package:类的包定义
java.lang.reflect 包下主要包含一些实现反射功能的工具类,实际上,java.lang.reflect 包所有提供的反射API扩充了读取运行时Annotation信息的能力。当一个Annotation类型被定义为运行时的Annotation后,该注解才能是运行时可见,当class文件被装载时被保存在class文件中的Annotation才会被虚拟机读取。
AnnotatedElement 接口是所有程序元素(Class、Method和Constructor)的父接口,所以程序通过反射获取了某个类的AnnotatedElement对象之后,程序就可以调用该对象的如下四个个方法来访问Annotation信息:
方法1: T getAnnotation(Class annotationClass):
返回改程序元素上存在的、指定类型的注解,如果该类型注解不存在,则返回null。
方法2:Annotation[] getAnnotations():返回该程序元素上存在的所有注解。
方法3:boolean is AnnotationPresent(Class<?extends Annotation> annotationClass):判断该程序元素上是否包含指定类型的注解,存在则返回true,否则返回false.
方法4:Annotation[] getDeclaredAnnotations():返回直接存在于此元素上的所有注释。与此接口中的其他方法不同,该方法将忽略继承的注释。(如果没有注释直接存在于此元素上,则返回长度为零的一个数组。)该方法的调用者可以随意修改返回的数组;这不会对其他调用者返回的数组产生任何影响。
实例
FruitName.java
import java.lang.annotation.*;
/**
* @Description 水果名称注解
* @Author liqinglong
* @DateTime 2024-05-14 11:08
* @Version 1.0
*/
@Documented
@Target(ElementType.FIELD)
@Retention(RetentionPolicy.RUNTIME)
public @interface FruitName {
String value() default "";
}
FruitColor.java
import java.lang.annotation.*;
/**
* @Description 水果颜色注解
* @Author liqinglong
* @DateTime 2024-05-14 11:12
* @Version 1.0
*/
@Documented
@Target(ElementType.FIELD)
@Retention(RetentionPolicy.RUNTIME)
public @interface FruitColor {
/**
* 颜色枚举
*/
enum Color{BULE,RED,GREEN};
/**
* @Description 颜色属性
* @return com.ruoyi.web.annotation.FruitColor.Color
*/
Color fruitColor() default Color.GREEN;
}
FruitProvider.java
import java.lang.annotation.*;
/**
* 水果供应商注解
*/
@Documented
@Target(ElementType.FIELD)
@Retention(RetentionPolicy.RUNTIME)
public @interface FruitProvider {
/**
* 供应商编号
* @return int
*/
public int id() default -1;
/**
* 供应商名称
* @return java.lang.String
*/
public String name() default "";
/**
* 供应商地址
* @return java.lang.String
*/
public String address() default "";
}
Apple.java
import lombok.Data;
/**
* @Description 苹果实体
* @Author liqinglong
* @DateTime 2024-05-14 11:16
* @Version 1.0
*/
@Data
public class Apple {
@FruitName("Apple")
private String appleName;
@FruitColor(fruitColor = FruitColor.Color.RED)
private String appleColor;
@FruitProvider(id = 1,name = "贵州酒店集团",address = "贵州饭店")
private String appleProvider;
}
注解处理工具类
FruitInfoUtil.java
import java.lang.reflect.Field;
/**
* @Description TODO
* @Author liqinglong
* @DateTime 2024-05-14 11:55
* @Version 1.0
*/
public class FruitInfoUtil {
public static void getFruitInfo(Class<?> clazz){
String strFruitName=" 水果名称:";
String strFruitColor=" 水果颜色:";
String strFruitProvicer="供应商信息:";
Field[] fields = clazz.getDeclaredFields();
for(Field field :fields){
if(field.isAnnotationPresent(FruitName.class)){
FruitName fruitName = (FruitName) field.getAnnotation(FruitName.class);
strFruitName=strFruitName+fruitName.value();
System.out.println(strFruitName);
}
else if(field.isAnnotationPresent(FruitColor.class)){
FruitColor fruitColor= (FruitColor) field.getAnnotation(FruitColor.class);
strFruitColor=strFruitColor+fruitColor.fruitColor().toString();
System.out.println(strFruitColor);
}
else if(field.isAnnotationPresent(FruitProvider.class)){
FruitProvider fruitProvider= (FruitProvider) field.getAnnotation(FruitProvider.class);
strFruitProvicer=" 供应商编号:"+fruitProvider.id()+" 供应商名称:"+fruitProvider.name()+" 供应商地址:"+fruitProvider.address();
System.out.println(strFruitProvicer);
}
}
}
}
测试类
/**
* @Description TODO
* @Author liqinglong
* @DateTime 2024-05-14 14:55
* @Version 1.0
*/
public class FruitRun {
public static void main(String[] args) {
Apple apple = new Apple();
FruitInfoUtil.getFruitInfo(apple.getClass());
}
}
运行结果
总结
至此Java注解基础知识点咱们了解得基本差不多了,若想更深入的了解,需要我们不断练习和使用它,在下一篇我们通过设计一个基于注解的简单的ORM框架,来综合应用和进一步加深对注解的各个知识点的理解和运用。