注解的应用

1、简介

1.1 概念

Java 注解(Annotation)又称 Java 标注,是 JDK5.0 引入的一种注释机制。
Java 语言中的类、构造器、方法、成员变量、参数等都可以被注解进行标注。

1.2 作用

对Java中类、方法、成员变量做标记,然后进行特殊处理,至于到底做何种处理由业务需求来决定。
例如:JUnit框架中,标记了注解@Test的方法就可以被当成测试方法执行,而没有标记的就不能当成测试方法执行。

2、自定义注解

2.1 自定义注解格式

public @interface 注解名称 {
    public 属性类型 属性名() default 默认值 ;
}

2.2 特殊属性

value属性,如果只有一个value属性的情况下,使用value属性的时候可以省略value名称不写!!
但是如果有多个属性, 且多个属性没有默认值,那么value名称是不能省略的。

package com.annotation;
public @interface Book {
    String value();
}

2.3 自定义注解格式示例

package com.annotation;
public @interface MyBook {

    public String name();
    String[] authors();
    double price();
}
package com.annotation;
public @interface Book {
    String value();
}
package com.annotation;
public @interface TestBook {
    String value();
    double price() default 12;
}
package com.annotation;
//有多个属性,  其他属性都有默认值,那么value名称可以省略
@TestBook("222")
//只有一个属性value的情况下,使用value属性的时候可以省略,名称不写
@Book("111")
//有多个属性,  且多个属性没有默认值,那么value名称是不能省略的
@MyBook(name = "精通java", authors = {"张三"}, price = 100)
public class AnnotationDemo {

    @MyBook(name = "精通java", authors = {"张三"}, price = 100)
    public AnnotationDemo() {
    }

    @MyBook(name = "精通java", authors = {"张三"}, price = 100)
    public static void main(String[] args) {

        @MyBook(name = "精通java", authors = {"张三"}, price = 100)
        int age = 10;
    }
}

3、元注解

3.1 元注解是什么

注解注解的注解。

3.1 常见的两个元注解

@Target: 约束自定义注解只能在哪些地方使用,可使用的值定义在ElementType枚举类中

枚举值说明
ElementType.TYPE类,接口
ElementType.FIELD成员变量
ElementType.METHOD成员方法
ElementType.PARAMETER方法参数
ElementType.CONSTRUCTOR构造器
ElementType.LOCAL_VARIABLE局部变量

@Retention:申明注解的生命周期,可使用的值定义在RetentionPolicy枚举类中

枚举值说明
RetentionPolicy.SOURCE注解只作用在源码阶段,生成的字节码文件中不存在
RetentionPolicy.CLASS注解作用在源码阶段,字节码文件阶段,运行阶段不存在,默认值.
RetentionPolicy.RUNTIME注解作用在源码阶段,字节码文件阶段,运行阶段(开发常用
package com.annotation;
import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;
import java.lang.reflect.Field;

//只能注解方法和成员变量
@Target({ElementType.METHOD, ElementType.FIELD})
//一直活着,在运行阶段这个注解也不消失
@Retention(RetentionPolicy.RUNTIME)
public @interface MyTest {
}

package com.annotation;

//@MyTest  不能注解类
public class AnnotationDemo02 {
    @MyTest
    private String name;

    @MyTest
    public static void main(String[] args) {

    }
}

4、注解的解析

4.1 注解解析是什么

注解的操作中经常需要进行解析,注解的解析就是判断是否存在注解,存在注解就解析出内容。

4.2 与注解解析相关的接口

Annotation: 注解的顶级接口,注解都是Annotation类型的对象
AnnotatedElement:该接口定义了与注解解析相关的解析方法

方法说明
Annotation[] getDeclaredAnnotations()获得当前对象上使用的所有注解,返回注解数组
T getDeclaredAnnotation(Class annotationClass)根据注解类型获得对应注解对象
boolean isAnnotationPresent(Class annotationClass)判断当前对象是否使用了指定的注解,如果使用了则返回true,否则false

所有的类成分Class, Method , Field , Constructor,都实现了AnnotatedElement接口他们都拥有解析注解的能力

4.3 解析注解的技巧

注解在哪个成分上,我们就先拿哪个成分对象。
比如注解作用成员方法,则要获得该成员方法对应的Method对象,再来拿上面的注解
比如注解作用在类上,则要该类的Class对象,再来拿上面的注解
比如注解作用在成员变量上,则要获得该成员变量对应的Field对象,再来拿上面的注解

package com.annotatedelement;
import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;
//设置注解的使用范围为类和方法
@Target({ElementType.TYPE,ElementType.METHOD})
//设置注解作用在源码阶段,字节码文件阶段,运行阶段
@Retention(RetentionPolicy.RUNTIME)
public @interface BookTest {
    String value();
    double price() default 100;
    String[] authors();

}

package com.annotatedelement;
@BookTest(value = "书名1",authors = {"琼瑶","金庸1"})
public class BookStore {

    @BookTest(value = "书名2",authors = {"古龙","金庸2"})
    public void test(){

    }
}

package com.annotatedelement;
import org.junit.Test;
import java.lang.reflect.Method;
import java.util.Arrays;

/*
注解的解析
 */
public class AnnotatedElement01 {
    @Test
    public  void paraseClass() throws NoSuchMethodException {

        //得到类对象
        Class c = BookStore.class;

        //判断这个类是否有这个注解
        if(c.isAnnotationPresent(BookTest.class)){
            //获取注解对象
            BookTest book = (BookTest) c.getDeclaredAnnotation(BookTest.class);
            System.out.println(book.value());
            System.out.println(book.price());
            System.out.println(Arrays.toString(book.authors()));
        }
        //判断这个类的方法是否有这个注解
		Method m = c.getDeclaredMethod("test");
        if(m.isAnnotationPresent(BookTest.class)){
            //获取注解对象
            BookTest book = (BookTest) m.getDeclaredAnnotation(BookTest.class);
            System.out.println(book.value());
            System.out.println(book.price());
            System.out.println(Arrays.toString(book.authors()));
        }

    }
}

5、注解的应用场景:Junit框架

模拟Junit

package com.annotation;
import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;
import java.lang.reflect.Field;
//下面这部分为什么这样设置可以看Junit的@Test源码
//只能注解方法
@Target({ElementType.METHOD})
//一直活着,在运行阶段这个注解也不消失
@Retention(RetentionPolicy.RUNTIME)
public @interface MyTest {
}

package com.annotation;

import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;

public class AnnotationDemo03 {
    @MyTest
    public void test1(){
        System.out.println("test1");
    }
    public void test2(){
        System.out.println("test2");
    }
    @MyTest
    public void test3(){
        System.out.println("test3");
    }

    /*
    启动菜单:执行有注解的
     */
    public static void main(String[] args) throws InvocationTargetException, IllegalAccessException {
        AnnotationDemo03 t = new AnnotationDemo03();
        //获取类对象
        Class c  = AnnotationDemo03.class;
        //提取全部方法
        Method[] methods = c.getDeclaredMethods();
        //遍历方法,有注解就执行
        for (Method method : methods) {
            if(method.isAnnotationPresent(MyTest.class)){
                //运行
                method.invoke(t);
            }
        }
    }
}

测试结果

test1
test3

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值