Java——注解/注释

本文详细介绍了Java注解的使用,包括自定义注解、JDK内置的注解如`@Override`和`@Deprecated`,元注解的作用,以及注解中的属性类型和数组。还探讨了注解的保留策略,如`@Retention`,并展示了如何通过反射机制获取注解信息。最后,讨论了注解在实际开发中的应用场景,如确保类中存在特定属性的需求。
摘要由CSDN通过智能技术生成

目录

注解概述

JDK内置的注解

元注解

Override注解 

Deprecated

注解中自定义属性

value();

注解中的属性类型

属性是一个数组

Retention的源代码

反射注解

通过反射机制获取属性对象的值 

通过反射机制获取注释信息


注解概述

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

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

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

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

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

        第一:注解使用时的语法格式是:
            @注解类型名
        
        第二:注解可以出现在类上、属性上、方法上、变量上等....
        注解还可以出现在注解类型上。

自定义注解:MyAnnotation 


public @interface MyAnnotation {
    // ??????
}

@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 OtherAnnotation {
}

默认情况下,注解可以出现在任意位置。


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文件中,并且可以被反射机制所读取。


Override注解 

源码:

@Target(ElementType.METHOD)
@Retention(RetentionPolicy.SOURCE)
public @interface Override {
}

@Override

标识性注解,给编译器做参考的。
编译器看到方法上有这个注解的时候,编译器会自动检查该方法是否重写了父类的方法。
如果没有重写,报错。

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

// @Override这个注解只能注解方法。
// @Override这个注解是给编译器参考的,和运行阶段没有关系
// 凡是java中的方法带有这个注解的,编译器都会进行编译检查,如果这个方法不是重写父类的方法,编译器报错。

Deprecated

Deprecated这个注解标注的元素已过时。
这个注解主要是向其它程序员传达一个信息,告知已过时,有更好的解决方案存在。

// 表示这个类已过时。
@Deprecated
public class AnnotationTest03 {

    @Deprecated
    private String s;

    public static void main(String[] args) {
        AnnotationTest03 at = new AnnotationTest03();
        at.doSome();
    }

    @Deprecated
    public void doSome(){
        System.out.println("do something!");
    }

    // Deprecated这个注解标注的元素已过时。
    // 这个注解主要是向其它程序员传达一个信息,告知已过时,有更好的解决方案存在。
    @Deprecated
    public static void doOther(){
        System.out.println("do other...");
    }
}

class T {
    public static void main(String[] args) {
        AnnotationTest03 at = new AnnotationTest03();
        at.doSome();

        AnnotationTest03.doOther();

        try {
            Class c = Class.forName("java.util.Date");
            Object obj = c.newInstance();
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}

注解中自定义属性

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 = "红色")
    public void doSome(){

    }
}

value();

如果一个注解的属性的名字是value,并且只有一个属性的话,在使用的时候,该属性名可以省略。

public @interface MyAnnotation {

    /*
    指定一个value属性。
     */
    String value();

    //String email();
}
public class MyAnnotationTest {

    @MyAnnotation(value = "hehe")
    public void doSome(){

    }

    @MyAnnotation("haha")
    public void doOther(){

    }
}

注解中的属性类型

注解当中的属性可以是哪一种类型?
        属性的类型可以是:
            byte short int long float double boolean char String Class 枚举类型
            以及以上每一种的数组形式。

public @interface MyAnnotation {

    int value1();

    String value2();

    int[] value3();

    String[] value4();

    Season value5();

    Season[] value6();

    Class parameterType();

    Class[] parameterTypes();
}
public enum Season {
    SPRING,SUMMER,AUTUMN,WINTER
}

属性是一个数组

    // 数组是大括号
    @OtherAnnotation(age = 25, email = {"zhangsan@123.com", "zhangsan@sohu.com"}, seasonArray = Season.WINTER)
    public void doSome(){

    }

    // 如果数组中只有1个元素:大括号可以省略。
    @OtherAnnotation(age = 25, email = "zhangsan@123.com", seasonArray = {Season.SPRING, Season.SUMMER})
    public void doOther(){

    }

Retention的源代码

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

            //@Retention(value=RetentionPolicy.RUNTIME)
            @Retention(RetentionPolicy.RUNTIME)
            public @interface MyAnnotation{}


反射注解

通过反射机制获取属性对象的值 

//只允许该注解可以标注类、方法
@Target({ElementType.TYPE, ElementType.METHOD})
// 希望这个注解可以被反射
@Retention(RetentionPolicy.RUNTIME)
public @interface MyAnnotation {

    /*
    value属性。
     */
    String value() default "沈阳大街";
}
//Test

@MyAnnotation("山东菏泽曹县")
public class MyAnnotationTest {

    //@MyAnnotation
    int i;

    //@MyAnnotation
    public MyAnnotationTest(){

    }

    @MyAnnotation
    public void doSome(){

        //@MyAnnotation
        int i;
    }
    public static void main(String[] args) throws Exception{
        // 获取这个类
        Class c = Class.forName("lit.jxxy.rg1508.java.annotation5.MyAnnotationTest");
        // 判断类上面是否有@MyAnnotation
        //System.out.println(c.isAnnotationPresent(MyAnnotation.class)); // true
        if(c.isAnnotationPresent(MyAnnotation.class)){
            // 获取该注解对象
            MyAnnotation myAnnotation = (MyAnnotation)c.getAnnotation(MyAnnotation.class);
            //System.out.println("类上面的注解对象" + myAnnotation); // @lit.jxxy.rg1508.java.annotation5.MyAnnotation()
            // 获取注解对象的属性怎么办?和调接口没区别。
            String value = myAnnotation.value();
            System.out.println(value);
        }

        // 判断String类上面是否存在这个注解
        Class stringClass = Class.forName("java.lang.String");
        System.out.println(stringClass.isAnnotationPresent(MyAnnotation.class)); // false
    }

通过反射机制获取注释信息

@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.METHOD)
public @interface MyAnnotation {

    /*
    username属性
     */
    String username();

    /*
    password属性
     */
    String password();
}

    @MyAnnotation(username = "admin", password = "456456")
    public void doSome(){

    }

    public static void main(String[] args) throws Exception{
        // 获取MyAnnotationTest的doSome()方法上面的注解信息。
        Class c = Class.forName("lit.jxxy.rg1508.java.annotation6.MyAnnotationTest");
        // 获取doSome()方法
        Method doSomeMethod = c.getDeclaredMethod("doSome");
        // 判断该方法上是否存在这个注解
        if(doSomeMethod.isAnnotationPresent(MyAnnotation.class)) {
            MyAnnotation myAnnotation = doSomeMethod.getAnnotation(MyAnnotation.class);
            System.out.println(myAnnotation.username());
            System.out.println(myAnnotation.password());
        }
    }

注解在开发中的作用

     需求:
            假设有这样一个注解,叫做:@Id
            这个注解只能出现在类上面,当这个类上有这个注解的时候,
            要求这个类中必须有一个int类型的id属性。如果没有这个属性
            就报异常。如果有这个属性则正常执行!

// 表示这个注解只能出现在类上面
@Target(ElementType.TYPE)
// 该注解可以被反射机制读取到
@Retention(RetentionPolicy.RUNTIME)
public @interface MustHasIdPropertyAnnotation {

}

// 这个注解@Id用来标注类,被标注的类中必须有一个int类型的id属性,没有就报异常。
@MustHasIdPropertyAnnotation
public class User {
    int id;
    String name;
    String password;
}
/*
自定义异常
 */
public class HasNotIdPropertyException extends RuntimeException {
    public HasNotIdPropertyException(){

    }
    public HasNotIdPropertyException(String s){
        super(s);
    }
}
    public static void main(String[] args) throws Exception{
        // 获取类
        Class userClass = Class.forName("lit.jxxy.rg1508.java.annotation7.User");
        // 判断类上是否存在Id注解
        if(userClass.isAnnotationPresent(MustHasIdPropertyAnnotation.class)){
            // 当一个类上面有@MustHasIdPropertyAnnotation注解的时候,要求类中必须存在int类型的id属性
            // 如果没有int类型的id属性则报异常。
            // 获取类的属性
            Field[] fields = userClass.getDeclaredFields();
            boolean isOk = false; // 给一个默认的标记
            for(Field field : fields){
                if("id".equals(field.getName()) && "int".equals(field.getType().getSimpleName())){
                    // 表示这个类是合法的类。有@Id注解,则这个类中必须有int类型的id
                    isOk = true; // 表示合法
                    break;
                }
            }

            // 判断是否合法
            if(!isOk){
                throw new HasNotIdPropertyException("被@MustHasIdPropertyAnnotation注解标注的类中必须要有一个int类型的id属性!");
            }

        }
    }

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值