zoukankan      html  css  js  c++  java
  • Java注解学习

    一、注解定义

    JVM5.0定义了4个标准的元注解:

    • @Target,
    • @Retention,
    • @Documented
    • @Inherited

    1. @Target

    作用:用于描述注解的使用范围
    取值ElementType有:

    • CONSTRUCTOR:用于描述构造器
    • FIELD:用于描述域
    • LOCAL_VARIABLE:用于描述局部变量
    • METHOD:用于描述方法
    • PACKAGE:用于描述包
    • PARAMETER:用于描述参数
    • TYPE:用于描述类、接口(包括注解类型) 或enum声明

    举例:

    @Target(ElementType.TYPE)
    public @interface Table {
        /**
         * 数据表名称注解,默认值为类名称
         * @return
         */
        public String tableName() default "className";
    }
    
    @Target(ElementType.FIELD)
    public @interface NoDBColumn {
    
    }
    

    注解Table可以用于注解类、接口(包括注解类型)或enum声明,而注解NoDBColumn仅用于注解类的成员变量。

    2. @Retention

    作用:用于描述注解的生命周期
    取值RetentionPolicy有:

    • SOURCE:在源文件中有效(即源文件保留)
    • CLASS:在class文件中有效(即class保留)
    • RUNTIME:在运行时有效(即运行时保留)

    举例:

    @Target(ElementType.FIELD)
    @Retention(RetentionPolicy.RUNTIME)
    public @interface Column {
        public String name() default "fieldName";
        public String setFuncName() default "setField";
        public String getFuncName() default "getField"; 
        public boolean defaultDBValue() default false;
    }
    
    

    Column注解的的RetentionPolicy的属性值是RUNTIME,这样注解处理器可以通过反射,获取到该注解的属性值,从而去做一些运行时的逻辑处理

    3. Documented

    作用:用于描述其它类型的annotation应该被作为被标注的程序成员的公共API,因此可以被例如javadoc此类的工具文档化。Documented是一个标记注解,没有成员。

    举例:

    @Target(ElementType.FIELD)
    @Retention(RetentionPolicy.RUNTIME)
    @Documented
    public @interface Column {
        public String name() default "fieldName";
        public String setFuncName() default "setField";
        public String getFuncName() default "getField"; 
        public boolean defaultDBValue() default false;
    }
    
    
    

    4. @Inherited

    @Inherited 元注解是一个标记注解,@Inherited阐述了某个被标注的类型是被继承的。如果一个使用了Inherited修饰的annotation类型被用于一个class,则这个annotation将被用于该class的子类。

    注意:Inherited annotation类型是被标注过的class的子类所继承。类并不从它所实现的接口继承annotation,方法并不从它所重载的方法继承annotation。

    当Inherited annotation类型标注的annotation的Retention是RetentionPolicy.RUNTIME,则反射API增强了这种继承性。如果我们使用java.lang.reflect去查询一个Inherited annotation类型的annotation时,反射代码检查将展开工作:检查class和其父类,直到发现指定的annotation类型被发现,或者到达类继承结构的顶层

    实例代码:

    @Inherited
    public @interface Greeting {
        public enum FontColor{ BULE,RED,GREEN};
        String name();
        FontColor fontColor() default FontColor.GREEN;
    }
    
    

    5. 自定义注解

    • 使用interface自定义注解,自动继承java.lang.annotation.Annotation接口。
    • 不能继承其他的注解或接口。
    • 每一个方法实际上是声明了一个配置参数。方法的名称就是参数的名称,返回值类型就是参数的类型(只能是基本类型、Class、String、enum)。可以通过default声明参数的默认值。

    定义注解格式:

    public @interface 注解名(定义体)

    注解参数的可支持数据类型:

    • 所有基本数据类型(int,float,boolean,byte,double,char,long,short)
    • String类型
    • Class类型
    • enum类型
    • Annotation类型
    • 以上所有类型的数组

    自定义注解的定义和具体实现可以见下一章节。

    二、注解实现

    创建注解处理器,利用反射对注解加以处理。

    AnnotatedElement 接口是所有程序元素(Class、Method和Constructor)的父接口,关于AnnotatedElement点详细信息可以参考

    JavaDoc:Interface AnnotatedElement

    所以程序通过反射获取了某个类的AnnotatedElement对象之后,程序就可以调用该对象的如下四个个方法来访问Annotation信息:

    • 方法1:
    <T extends Annotation> T getAnnotation(Class<T> annotationClass):
    

    返回改程序元素上存在的、指定类型的注解,如果该类型注解不存在,则返回null。

    • 方法2:Annotation[] getAnnotations():返回该程序元素上存在的所有注解。

    • 方法3:boolean is AnnotationPresent(Class<?extends Annotation> annotationClass):判断该程序元素上是否包含指定类型的注解,存在则返回true,否则返回false.

    • 方法4:Annotation[] getDeclaredAnnotations():返回直接存在于此元素上的所有注释。与此接口中的其他方法不同,该方法将忽略继承的注释。(如果没有注释直接存在于此元素上,则返回长度为零的一个数组。)该方法的调用者可以随意修改返回的数组;这不会对其他调用者返回的数组产生任何影响。

    举例:

    注解声明

    1.FruitName

    package annotation;
    
    import java.lang.annotation.Documented;
    import java.lang.annotation.ElementType;
    import java.lang.annotation.Retention;
    import java.lang.annotation.RetentionPolicy;
    import java.lang.annotation.Target;
    
    @Target(ElementType.FIELD)
    @Retention(RetentionPolicy.RUNTIME)
    @Documented
    public @interface FruitName {
    	String value() default "";
    }
    

    2.FruitColor

    package annotation;
    
    import java.lang.annotation.Documented;
    import java.lang.annotation.ElementType;
    import java.lang.annotation.Retention;
    import java.lang.annotation.RetentionPolicy;
    import java.lang.annotation.Target;
    
    @Target(ElementType.FIELD)
    @Retention(RetentionPolicy.RUNTIME)
    @Documented
    public @interface FruitColor {
    	public enum Color{BlUE, RED, GREEN};
    	Color fruitColor() default Color.GREEN;
    
    }
    
    

    3.FruitProvider

    package annotation;
    
    import java.lang.annotation.Documented;
    import java.lang.annotation.ElementType;
    import java.lang.annotation.Retention;
    import java.lang.annotation.RetentionPolicy;
    import java.lang.annotation.Target;
    
    @Target(ElementType.FIELD)
    @Retention(RetentionPolicy.RUNTIME)
    @Documented
    public @interface FruitProvider {
    	public int id() default -1;
    	public String name() default "";
    	public String address() default "";
    }
    
    

    注解使用

    package annotation;
    
    import annotation.FruitColor.Color;
    
    public class Apple {
    	@FruitName("Apple")
    	private String appleName;
    	
    	@FruitColor(fruitColor=Color.RED)
    	private String appleColor;
    
    	@FruitProvider(id=1, name="红富士集团", address="红富士大厦")
    	private String appleProvider;
    
    	public String getAppleName() {
    		return appleName;
    	}
    
    	public void setAppleName(String appleName) {
    		this.appleName = appleName;
    	}
    
    	public String getAppleColor() {
    		return appleColor;
    	}
    
    	public void setAppleColor(String appleColor) {
    		this.appleColor = appleColor;
    	}
    	
    	public String getAppleProvider() {
    		return appleProvider;
    	}
    
    	public void setAppleProvider(String appleProvider) {
    		this.appleProvider = appleProvider;
    	}
    	
    	public void display(){
    		System.out.println("The fruit name is:" + appleName + ", its color is: " + appleColor + ".");
    	}
    	
    }
    
    

    注解实现

    package annotation;
    
    import java.lang.reflect.Field;
    
    public class FruitInfoUtil {
    	public static void getFruitInfo(Class<?> clazz){
    		String strFruitName= "水果名称: ";
    		String strFruitColor= "水果颜色: ";
    		String strFruitProvider= "供应商信息: ";
    		
    		Field[] fields = clazz.getDeclaredFields();
    		
    		for(Field field : fields){
    			if (field.isAnnotationPresent(FruitName.class)){
    				FruitName fruitName = (FruitName)field.getAnnotation(FruitName.class);
    				strFruitName += fruitName.value();
    				System.out.println(strFruitName);
    			}
    			
    			
    			if (field.isAnnotationPresent(FruitColor.class)){
    				FruitColor fruitColor = (FruitColor)field.getAnnotation(FruitColor.class);
    				strFruitColor += fruitColor.fruitColor().toString();
    				System.out.println(strFruitColor);
    			}
    			
    			if (field.isAnnotationPresent(FruitProvider.class)){
    				FruitProvider fruitProvider = (FruitProvider)field.getAnnotation(FruitProvider.class);
    				strFruitProvider += " 供应商编号:" + fruitProvider.id() + " 供应商名称:" + fruitProvider.name() + " 供应商地址:" + fruitProvider.address();
    				System.out.println(strFruitProvider);
    			}
    		}
    	}
    }
    
    

    执行结果:

    水果名称: Apple
    水果颜色: RED
    供应商信息:  供应商编号:1 供应商名称:红富士集团 供应商地址:红富士大厦
    

    三、注解基础知识点汇总

    注解基础知识点汇总

    四、参考博客

    1. 深入理解Java:注解
    2. Java技术之反射
    3. Java技术之注解
  • 相关阅读:
    MVVM
    vue-cli初始化项目2.x|3.x
    逻辑覆盖
    white box白盒测试
    black box黑盒测试
    总结回顾js arr的常见方法以及相关的使用场景(一)
    js 原生功底 (一)
    markdown 语法总结(一)
    阿里一面,面试官想看到的究竟是什么,带你揭秘!!!!
    关于Axios 源码你想了解的 在这儿
  • 原文地址:https://www.cnblogs.com/znicy/p/6852090.html
Copyright © 2011-2022 走看看