Java基础进阶注解

简介: Java基础进阶注解

注解,或者叫做注释类型,英文单词是:Annotation


注解Annotation是一种引用数据类型。编译之后也是生成xxx.class文件。


怎么自定义注解呢?语法格式?


[修饰符列表] @interface 注解类型名{
 }


注解怎么使用,用在什么地方?


第一:注解使用时的语法格式是:

@注解类型名


第二:注解可以出现在类上、属性上、方法上、变量上等…

注解还可以出现在注解类型上。


JDK内置了哪些注解呢?


java.lang包下的注释类型:


掌握:

Deprecated 用 @Deprecated 注释的程序元素,

不鼓励程序员使用这样的元素,通常是因为它很危险或存在更好的选择。


掌握:

Override 表示一个方法声明打算重写超类中的另一个方法声明。


不用掌握:

SuppressWarnings 指示应该在注释元素(以及包含在该注释元素中的

所有程序元素)中取消显示指定的编译器警告。


元注解

什么是元注解?


用来标注“注解类型”的“注解”,称为元注解。

常见的元注解有哪些?


Target

Retention

关于Target注解:


这是一个元注解,用来标注“注解类型”的“注解”

这个Target注解用来标注“被标注的注解”可以出现在哪些位置上。

@Target(ElementType.METHOD):表示“被标注的注解”只能出现在方法上。

@Target(value={CONSTRUCTOR, FIELD, LOCAL_VARIABLE, METHOD, PACKAGE, MODULE, PARAMETER, TYPE})

表示该注解可以出现在:

构造方法上

字段上

局部变量上

方法上

类上…

关于Retention注解:


这是一个元注解,用来标注“注解类型”的“注解”

这个Retention注解用来标注“被标注的注解”最终保存在哪里。

@Retention(RetentionPolicy.SOURCE):表示该注解只被保留在java源文件中。

@Retention(RetentionPolicy.CLASS):表示该注解被保存在class文件中。

@Retention(RetentionPolicy.RUNTIME):表示该注解被保存在class文件中,并且可以被反射机制所读取。


Retention的源代码


//元注解 
public @interface Retention {
//属性
  RetentionPolicy value();
}
RetentionPolicy的源代码:
  public enum RetentionPolicy {
   SOURCE,
   CLASS,
   RUNTIME
  }
//@Retention(value=RetentionPolicy.RUNTIME)
@Retention(RetentionPolicy.RUNTIME)
public @interface MyAnnotation{}


Target的源代码


public @interface Target {
    /**
     * Returns an array of the kinds of elements an annotation type
     * can be applied to.
     * @return an array of the kinds of elements an annotation type
     * can be applied to
     */
    ElementType[] value();
}
public enum ElementType {
    /** Class, interface (including annotation type), or enum declaration */
    TYPE,
    /** Field declaration (includes enum constants) */
    FIELD,
    /** Method declaration */
    METHOD,
    /** Formal parameter declaration */
    PARAMETER,
    /** Constructor declaration */
    CONSTRUCTOR,
    /** Local variable declaration */
    LOCAL_VARIABLE,
    /** Annotation type declaration */
    ANNOTATION_TYPE,
    /** Package declaration */
    PACKAGE,
    /**
     * Type parameter declaration
     *
     * @since 1.8
     */
    TYPE_PARAMETER,
    /**
     * Use of a type
     *
     * @since 1.8
     */
    TYPE_USE
}


示例代码01:


// 默认情况下,注解可以出现在任意位置。
@MyAnnotation
public class AnnotationTest01 {
    @MyAnnotation
    private int no;
    @MyAnnotation
    public AnnotationTest01(){}
    @MyAnnotation
    public static void m1(){
        @MyAnnotation
        int i = 100;
    }
    @MyAnnotation
    public void m2(@MyAnnotation
                   String name,
                   @MyAnnotation
                   int k){
    }
}
@MyAnnotation
interface MyInterface {
}
@MyAnnotation
enum Season {
    SPRING,SUMMER,AUTUMN,WINTER
}
/*
自定义注解:MyAnnotation
 */
public @interface MyAnnotation {
    // ??????
}
// 注解修饰注解。
@MyAnnotation
public @interface OtherAnnotation {
}


标识性注解,给编译器做参考的。

编译器看到方法上有这个注解的时候,编译器会自动检查该方法是否重写了父类的方法。

如果没有重写,报错。


这个注解只是在编译阶段起作用,和运行期无关!


示例代码02:


// @Override这个注解只能注解方法。
// @Override这个注解是给编译器参考的,和运行阶段没有关系。
// 凡是java中的方法带有这个注解的,编译器都会进行编译检查,如果这个方法不是重写父类的方法,编译器报错。
//@Override
public class AnnotationTest02 {
    //@Override
    private int no;
    @Override
    public String toString() {
        return "toString";
    }
}


示例代码03:


// 表示这个类已过时。
@Deprecated
public class AnnotationTest03 {
    @Deprecated
    private int no;
    @Deprecated
    public void doSome() {
        System.out.println("doSome....");
    }
    // Deprecated这个注解标注的元素已过时。
    // 这个注解主要是向其它程序员传达一个信息,告知已过时,有更好的解决方案存在。
    @Deprecated
    public static void doOther() {
        System.out.println("doOther...");
    }
}
class T{
    public static void main(String[] args) throws ClassNotFoundException {
        AnnotationTest03 an3 = new AnnotationTest03();
        an3.doSome();
        AnnotationTest03.doOther();
    }


运行结果:


0a2653c851af460fa595bd959398a8f1.png


示例代码04:


public @interface MyAnnotation {
    /**
     * 我们通常在注解当中可以定义属性,以下这个是MyAnnotation的name属性。
     * 看着像1个方法,但实际上我们称之为属性name。
     * @return
     */
    String name();
    /*
    颜色属性
     */
    String color();
    /*
    年龄属性
     */
    int age() default 25;//属性指定默认值
}
public class MyAnnotationTest {
    // 报错的原因:如果一个注解当中有属性,那么必须给属性赋值。(除非该属性使用default指定了默认值。)
    /*@MyAnnotation
    public void doSome(){
    }*/
    //@MyAnnotation(属性名=属性值,属性名=属性值,属性名=属性值)
    //指定name属性的值就好了。
    @MyAnnotation(name="zhangsan",color="red")
    public void doSome(){
    }
}


示例代码05:


public @interface MyAnnotation {
    /*
   指定一个value属性。
    */
    String value();
    //String email();
}
/*
如果一个注解的属性的名字是value,并且只有一个属性的话,在使用的时候,该属性名可以省略。
 */
public class MyAnnotationTest {
    // 报错原因:没有指定属性的值。
    /*@MyAnnotation
    public void doSome(){
    }*/
    @MyAnnotation(value = "hehe")
    public void doSome(){
    }
    @MyAnnotation("haha")
    public void doOther(){
    }
}
=====================================================================================
public @interface OtherAnnotation {
    String name();
}
// 报错了。因为属性名是name,不能省略。
//@OtherAnnotation("test")
public class OtherAnnotationTest {
    /*@OtherAnnotation(value="hh")
    public void doSome(){
    }*/
    // 正确的。
    @OtherAnnotation(name="test")
    public void doSome(){
    }
}


示例代码06:


public @interface MyAnnotion {
    /*
    注解当中的属性可以是哪一种类型?
        属性的类型可以是:
            byte short int long float double boolean char String Class 枚举类型
            以及以上每一种的数组形式。
     */
    int value1();
    int[] value6();
    String value2();
    String[] value3();
    Season value4();
    Season[] value5();
    Class value7();
    Class[] value8();
}
public @interface MyAnnotion {
    /*
    注解当中的属性可以是哪一种类型?
        属性的类型可以是:
            byte short int long float double boolean char String Class 枚举类型
            以及以上每一种的数组形式。
     */
    int value1();
    int[] value6();
    String value2();
    String[] value3();
    Season value4();
    Season[] value5();
    Class value7();
    Class[] value8();
}
public enum Season {
    SPRING,SUMMER,AUTUMN,WINTER
}
=================================================================================
public @interface OtherAnnotation {
    /**
     * 字符串数组,name数组
     * @return
     */
    String[] name();
    /**
     * 季节数组,Season是枚举类型
     * @return
     */
    Season[] season();
}
public class OtherAnnotationTest {
    // 数组是大括号
    @OtherAnnotation(name={"zahngsan","lisi","wangwu"},season = {Season.SPRING,Season.AUTUMN})
    public void doSome(){
    }
    // 数组是大括号
    @OtherAnnotation(name="zhangsan",season = Season.WINTER)
    public void doOther(){
    }
}


示例代码07:


//元注解
@Target({ElementType.TYPE,ElementType.METHOD})//定义此注解只能出现在类和方法上
@Retention(RetentionPolicy.RUNTIME)//定义此注解保存在字节码(class)文件中,并且能被反射机制读取到
@interface MyAnnotation {
    String value() default "广东深圳";
}
@MyAnnotation("广东省")
class MyAnnotationTest {
    @MyAnnotation("广东省")
    public void doSome(){
    }
}
//通过反射获取注解的属性值
public class ReflectAnnotationTest {
    public static void main(String[] args) throws ClassNotFoundException {
        //获取到MyAnnotation类
        Class c = Class.forName("annotation5.MyAnnotationTest");
        //判断类中是否存在某注解
        boolean flag = c.isAnnotationPresent(MyAnnotation.class);
        //如果存在,通过反射获取类的属性,并输出
        if(flag){
            MyAnnotation Myannotation = (MyAnnotation)c.getAnnotation(MyAnnotation.class);
            String value = Myannotation.value();
            System.out.println("该注解的属性名是:" + value);
        }
        System.out.println(flag);
        //判断字符串类中是否存在该注解
        Class stringclass = Class.forName("java.lang.String");
        boolean flag1 = stringclass.isAnnotationPresent(MyAnnotation.class);
        System.out.println(flag1);
    }
}


运行结果:


2d65d23f6d4748949b924e4057485923.png


示例代码08:


@Target({ElementType.TYPE,ElementType.METHOD})
@Retention(RetentionPolicy.RUNTIME)
@interface MyAnnotation {
    String username();
    String password();
}
public class MyAnnotationTest {
    @MyAnnotation(username="admin",password="123123")
    public void doSome(){
    }
    public static void main(String[] args) throws Exception{
        //获取MyAnnotationTest类
        Class c = Class.forName("annotation6.MyAnnotationTest");
        //获取类中的方法
        Method doSome = c.getDeclaredMethod("doSome");
        //判断doSome方法上是否有注解,如果有,就输出注解属性值
        if(doSome.isAnnotationPresent(MyAnnotation.class)){
            MyAnnotation Myannotation = doSome.getAnnotation(MyAnnotation.class);
            System.out.println(Myannotation.username());
            System.out.println(Myannotation.password());
        }
    }
}


运行结果:


6de278e6d6694ce5bb08e7e842b7e74b.png


需求:


假设有这样一个注解,叫做:@Id

这个注解只能出现在类上面,当这个类上有这个注解的时候,

要求这个类中必须有一个int类型的id属性。如果没有这个属性就报异常。

如果有这个属性则正常执行!


示例代码09:


//自定义异常
class HasntIdPropertyException extends RuntimeException{
    public HasntIdPropertyException(){
    }
    public HasntIdPropertyException(String s){
        super(s);
    }
}
public class Test {
    public static void main(String[] args) throws Exception{
        //获取类
        Class c = Class.forName("annotation7.User");
        //判断类中是否存在Id注解
        if(c.isAnnotationPresent(Id.class)){
            //获取类属性
            boolean isOk = false;// 给一个默认的标记
            // 当一个类上面有@MustHasIdPropertyAnnotation注解的时候,要求类中必须存在int类型的id属性
            // 如果没有int类型的id属性则报异常。
            // 获取类的属性
            Field[] fields = c.getDeclaredFields();
            for(Field field : fields) {
                if ("no".equals(field.getName()) && "int".equals(field.getType().getSimpleName())) {
                    // 表示这个类是合法的类。有@Id注解,则这个类中必须有int类型的id
                    isOk = true;// 表示合法
                    break;
                }
            }
            // 判断是否合法
                if(!isOk){
                    throw new HasntIdPropertyException("被@NustHasIdPropertyAnnotation注解标注的类中必须有一个int类型的的的Id属性!");
                }
            }
        }
    }
@Id
class User {
    String no;
    String name;
    int age;
}
// 表示这个注解只能出现在类上面
@Target({ElementType.FIELD,ElementType.TYPE,ElementType.METHOD})
// 该注解可以被反射机制读取到
@Retention(RetentionPolicy.RUNTIME)
@interface Id {
}


运行结果:


8ec4f2997fb246878c34ecd6d122b7c6.png12c3b7f3f8814309a195c64f051d4445.png

相关文章
|
26天前
|
XML Java 编译器
Java注解的底层源码剖析与技术认识
Java注解(Annotation)是Java 5引入的一种新特性,它提供了一种在代码中添加元数据(Metadata)的方式。注解本身并不是代码的一部分,它们不会直接影响代码的执行,但可以在编译、类加载和运行时被读取和处理。注解为开发者提供了一种以非侵入性的方式为代码提供额外信息的手段,这些信息可以用于生成文档、编译时检查、运行时处理等。
60 7
|
3月前
|
XML Java 编译器
Java学习十六—掌握注解:让编程更简单
Java 注解(Annotation)是一种特殊的语法结构,可以在代码中嵌入元数据。它们不直接影响代码的运行,但可以通过工具和框架提供额外的信息,帮助在编译、部署或运行时进行处理。
104 43
Java学习十六—掌握注解:让编程更简单
|
1月前
|
Java 编译器 数据库
Java 中的注解(Annotations):代码中的 “元数据” 魔法
Java注解是代码中的“元数据”标签,不直接参与业务逻辑,但在编译或运行时提供重要信息。本文介绍了注解的基础语法、内置注解的应用场景,以及如何自定义注解和结合AOP技术实现方法执行日志记录,展示了注解在提升代码质量、简化开发流程和增强程序功能方面的强大作用。
76 5
|
2月前
|
Java 开发者 Spring
[Java]自定义注解
本文介绍了Java中的四个元注解(@Target、@Retention、@Documented、@Inherited)及其使用方法,并详细讲解了自定义注解的定义和使用细节。文章还提到了Spring框架中的@AliasFor注解,通过示例帮助读者更好地理解和应用这些注解。文中强调了注解的生命周期、继承性和文档化特性,适合初学者和进阶开发者参考。
67 14
|
2月前
|
前端开发 Java
[Java]讲解@CallerSensitive注解
本文介绍了 `@CallerSensitive` 注解及其作用,通过 `Reflection.getCallerClass()` 方法返回调用方的 Class 对象。文章还详细解释了如何通过配置 VM Options 使自定义类被启动类加载器加载,以识别该注解。涉及的 VM Options 包括 `-Xbootclasspath`、`-Xbootclasspath/a` 和 `-Xbootclasspath/p`。最后,推荐了几篇关于 ClassLoader 的详细文章,供读者进一步学习。
37 12
|
3月前
|
存储 缓存 Java
java基础:IO流 理论与代码示例(详解、idea设置统一utf-8编码问题)
这篇文章详细介绍了Java中的IO流,包括字符与字节的概念、编码格式、File类的使用、IO流的分类和原理,以及通过代码示例展示了各种流的应用,如节点流、处理流、缓存流、转换流、对象流和随机访问文件流。同时,还探讨了IDEA中设置项目编码格式的方法,以及如何处理序列化和反序列化问题。
95 1
java基础:IO流 理论与代码示例(详解、idea设置统一utf-8编码问题)
|
2月前
|
Java 编译器
Java进阶之标准注解
Java进阶之标准注解
43 0
|
4月前
|
安全 Java API
【Java面试题汇总】Java基础篇——String+集合+泛型+IO+异常+反射(2023版)
String常量池、String、StringBuffer、Stringbuilder有什么区别、List与Set的区别、ArrayList和LinkedList的区别、HashMap底层原理、ConcurrentHashMap、HashMap和Hashtable的区别、泛型擦除、ABA问题、IO多路复用、BIO、NIO、O、异常处理机制、反射
|
3月前
|
JSON Java 数据库
java 常用注解大全、注解笔记
关于Java常用注解的大全和笔记,涵盖了实体类、JSON处理、HTTP请求映射等多个方面的注解使用。
50 0
java 常用注解大全、注解笔记
|
4月前
|
Arthas Java 测试技术
Java字节码文件、组成,jclasslib插件、阿里arthas工具,Java注解
Java字节码文件、组成、详解、分析;常用工具,jclasslib插件、阿里arthas工具;如何定位线上问题;Java注解
Java字节码文件、组成,jclasslib插件、阿里arthas工具,Java注解