Java注解

Java 注解

注解概述

3、注解
<span class="hljs-number">3.1</span>、注解,或者叫做注释类型,英文单词是:Annotation
	疑问:注解到底是干啥的?????????

<span class="hljs-number">3.2</span>、注解Annotation是一种引用数据类型。编译之后也是生成xxx.class文件。

<span class="hljs-number">3.3</span>、怎么自定义注解呢?语法格式?

	 [修饰符列表] <span class="hljs-meta">@interface</span> 注解类型名{

	 }

<span class="hljs-number">3.4</span>、注解怎么使用,用在什么地方?

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

<span class="hljs-number">3.5</span>、JDK内置了哪些注解呢?

	java.lang包下的注释类型:

		掌握:
		Deprecated 用 <span class="hljs-meta">@Deprecated</span> 注释的程序元素,表示该元素已过时。
		不鼓励程序员使用这样的元素,通常是因为它很危险或存在更好的选择。 

		掌握:
		Override 表示一个方法声明打算重写超类中的另一个方法声明。如果
        修改了重写的方法或者父类中没有该方法,那么使用该注解就会在编译阶段报错!

		不用掌握:
		SuppressWarnings 指示应该在注释元素(以及包含在该注释元素中的
		所有程序元素)中取消显示指定的编译器警告。 

<span class="hljs-number">3.6</span>、元注解
	什么是元注解?
		用来标注“注解类型”的“注解”,称为元注解。

	常见的元注解有哪些?
		Target
		Retention
	
	关于Target注解:
		这是一个元注解,用来标注“注解类型”的“注解”
		这个Target注解用来标注“被标注的注解”可以出现在哪些位置上。

		<span class="hljs-meta">@Target(ElementType.METHOD)</span>:表示“被标注的注解”只能出现在方法上。
		<span class="hljs-meta">@Target(value={CONSTRUCTOR, FIELD, LOCAL_VARIABLE, METHOD, PACKAGE, MODULE, PARAMETER, TYPE})</span>
			表示该注解可以出现在:
				构造方法上
				字段上
				局部变量上
				方法上
  			    包上
    			模块上
    			参数上
				....
				类上...
	
	关于Retention注解:
		这是一个元注解,用来标注“注解类型”的“注解”
		这个Retention注解用来标注“被标注的注解”最终保存在哪里。

		<span class="hljs-meta">@Retention(RetentionPolicy.SOURCE)</span>:表示该注解只被保留在java源文件中,编译之后没有这个注解。
		<span class="hljs-meta">@Retention(RetentionPolicy.CLASS)</span>:表示该注解被保存在class文件中。
		<span class="hljs-meta">@Retention(RetentionPolicy.RUNTIME)</span>:表示该注解被保存在class文件中,并且在运行时可以被反射机制所读取。
	
<span class="hljs-number">3.7</span>、Retention的源代码

		<span class="hljs-comment">//元注解	</span>
		<span class="hljs-keyword">public</span> <span class="hljs-meta">@interface</span> Retention {
			<span class="hljs-comment">//属性</span>
			RetentionPolicy <span class="hljs-title function_">value</span><span class="hljs-params">()</span>;
		}
		
	RetentionPolicy的源代码:
		<span class="hljs-keyword">public</span> <span class="hljs-keyword">enum</span> <span class="hljs-title class_">RetentionPolicy</span> {
			 SOURCE,
			 CLASS,
			 RUNTIME
		}

		<span class="hljs-comment">//@Retention(value=RetentionPolicy.RUNTIME)</span>
		<span class="hljs-meta">@Retention(RetentionPolicy.RUNTIME)</span>
		<span class="hljs-keyword">public</span> <span class="hljs-meta">@interface</span> MyAnnotation{}



<span class="hljs-number">3.8</span>、Target的源代码
	

<span class="hljs-number">3.9</span>、注解在开发中有什么用呢?

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

4、JDK新特性
后续。。。。。。。

一、如何定义和使用注解?

自定义注解:

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

}

// 自定义注解
public @interface MyAnnotation {
}

使用注解:如果没有使用 @Target 的话可以在任何位置上使用注解:

如:类 / 接口 / 枚举 / 注解 / 上。

属性、方法、方法形参上。

@Target 是什么注解,有什么用,有哪些属性值?

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

  • ANNOTATION_TYPE:注释类型声明
  • CONSTRUCTOR:构造方法声明
  • LOCAL_VARIABLE:局部变量声明
  • TYPE:类、接口(包括注释类型)或枚举声明
  • FIELD:字段声明(包括枚举常量)
  • METHOD:方法声明
  • PARAMETER:参数声明
  • PACKAGE:包声明
@MyAnnotation   // 出现在类上
public class Person {
    @MyAnnotation   // 出现在属性上
    private int no;
    @MyAnnotation
    private String name;
<span class="hljs-meta">@MyAnnotation</span>   <span class="hljs-comment">//出现在方法上</span>
<span class="hljs-keyword">public</span> <span class="hljs-title function_">Person</span><span class="hljs-params">()</span> {
}

<span class="hljs-meta">@MyAnnotation</span>
<span class="hljs-keyword">public</span> <span class="hljs-title function_">Person</span><span class="hljs-params">(<span class="hljs-meta">@MyAnnotation</span> <span class="hljs-type">int</span> no, <span class="hljs-meta">@MyAnnotation</span> String name)</span> {    <span class="hljs-comment">//出现在参数上</span>
    <span class="hljs-built_in">this</span>.no = no;
    <span class="hljs-built_in">this</span>.name = name;
}

<span class="hljs-meta">@MyAnnotation</span>
<span class="hljs-keyword">public</span> <span class="hljs-type">int</span> <span class="hljs-title function_">getNo</span><span class="hljs-params">()</span> {
    <span class="hljs-keyword">return</span> no;
}

<span class="hljs-meta">@MyAnnotation</span>
<span class="hljs-keyword">public</span> <span class="hljs-keyword">void</span> <span class="hljs-title function_">setNo</span><span class="hljs-params">(<span class="hljs-type">int</span> no)</span> {
    <span class="hljs-built_in">this</span>.no = no;
}

<span class="hljs-meta">@MyAnnotation</span>
<span class="hljs-keyword">public</span> String <span class="hljs-title function_">getName</span><span class="hljs-params">()</span> {
    <span class="hljs-keyword">return</span> name;
}

<span class="hljs-meta">@MyAnnotation</span>
<span class="hljs-keyword">public</span> <span class="hljs-keyword">void</span> <span class="hljs-title function_">setName</span><span class="hljs-params">(String name)</span> {
    <span class="hljs-built_in">this</span>.name = name;
}

}

二、JDK lang 包下的 Override 注解

Override 注解的特点:

  1. 该注解只能注解方法。
  2. 该注解给是给编译器参考的,和运行阶段没有关系。
  3. 凡是 java 中的方法带有这个注解的,编译器都会进行编译检查,如果这个方法不是重写父类的方法,编译器报错。
/*
关于 JDK lang 包下的 Override 注解
源代码:
public @interface Override {
}

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

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

*/

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

//@Override
public class AnnotationTest02 {

<span class="hljs-comment">//@Override</span>
<span class="hljs-keyword">private</span> <span class="hljs-type">int</span> no;

<span class="hljs-meta">@Override</span>
<span class="hljs-keyword">public</span> String <span class="hljs-title function_">toString</span><span class="hljs-params">()</span> {
    <span class="hljs-keyword">return</span> <span class="hljs-string">"toString"</span>;
}

}

三、@Deprecated 表示当前元素已过时

@Deprecated 可以用在类上、方法上、属性上... 表示当前元素已经过时。

@Deprecated 用在类上表示该类已经过时。

image-20200918202947288

image-20200918202732846

@Reprecated 用在方法上表示该方法已经过时。

image-20200918203136047

image-20200918203229896

@Reprecated 用在字段上表示该字段已经过时。

image-20200918203632288

// 表示这个类已过时。
@Deprecated
public class AnnotationTest03 {
<span class="hljs-meta">@Deprecated</span>
<span class="hljs-keyword">private</span> String s;

<span class="hljs-keyword">public</span> <span class="hljs-keyword">static</span> <span class="hljs-keyword">void</span> <span class="hljs-title function_">main</span><span class="hljs-params">(String[] args)</span> {
    <span class="hljs-type">AnnotationTest03</span> <span class="hljs-variable">at</span> <span class="hljs-operator">=</span> <span class="hljs-keyword">new</span> <span class="hljs-title class_">AnnotationTest03</span>();
    at.doSome();
}

<span class="hljs-meta">@Deprecated</span>
<span class="hljs-keyword">public</span> <span class="hljs-keyword">void</span> <span class="hljs-title function_">doSome</span><span class="hljs-params">()</span>{
    System.out.println(<span class="hljs-string">"do something!"</span>);
}

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

}

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

    AnnotationTest03.doOther();

    <span class="hljs-keyword">try</span> {
        <span class="hljs-type">Class</span> <span class="hljs-variable">c</span> <span class="hljs-operator">=</span> Class.forName(<span class="hljs-string">"java.util.Date"</span>);
        <span class="hljs-type">Object</span> <span class="hljs-variable">obj</span> <span class="hljs-operator">=</span> c.newInstance();
    } <span class="hljs-keyword">catch</span> (Exception e) {
        e.printStackTrace();
    }
}

}

四、注解中定义属性

重要结论:如果一个注解当中有属性,那么必须给属性赋值。(除非该属性使用 default 指定了默认值 )

@MyAnnotation(属性名 = 属性值, 属性名 = 属性值, 属性名 = 属性值)

MyAnnotation 注解:

public @interface MyAnnotation {
<span class="hljs-comment">/**
 * 我们通常在注解当中可以定义属性,以下这个是MyAnnotation的name属性。
 * 看着像1个方法,但实际上我们称之为属性name。
 * <span class="hljs-doctag">@return</span>
 */</span>
String <span class="hljs-title function_">name</span><span class="hljs-params">()</span>;

<span class="hljs-comment">/*
颜色属性
 */</span>
String <span class="hljs-title function_">color</span><span class="hljs-params">()</span>;

<span class="hljs-comment">/*
年龄属性
 */</span>
<span class="hljs-type">int</span> <span class="hljs-title function_">age</span><span class="hljs-params">()</span> <span class="hljs-keyword">default</span> <span class="hljs-number">25</span>; <span class="hljs-comment">//属性指定默认值</span>

}

MyAnnotationTest:测试

public class MyAnnotationTest {
<span class="hljs-comment">// 报错的原因:如果一个注解当中有属性,那么必须给属性赋值。(除非该属性使用default指定了默认值。)</span>
<span class="hljs-comment">/*@MyAnnotation
public void doSome(){

}*/</span>

<span class="hljs-comment">//@MyAnnotation(属性名=属性值,属性名=属性值,属性名=属性值)</span>
<span class="hljs-comment">//指定name属性的值就好了。</span>
<span class="hljs-meta">@MyAnnotation(name = "zhangsan", color = "红色")</span>
<span class="hljs-keyword">public</span> <span class="hljs-keyword">void</span> <span class="hljs-title function_">doSome</span><span class="hljs-params">()</span>{
}

}

五、注解中属性只有 value 时可以省略

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

注意:只有是属性名是 value 的时候才可以,如果是 name,哪怕该注解只有这一个属性,也不可能省略。

public @interface MyAnnotation {
<span class="hljs-comment">/*
指定一个value属性。
 */</span>
String <span class="hljs-title function_">value</span><span class="hljs-params">()</span>;

<span class="hljs-comment">//String email();	//在这要是再有个email,那value属性名就不能省略了。</span>

}

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

    // 报错原因:没有指定属性的值。
    /*@MyAnnotation
    public void doSome(){ }*/

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

    }

    @MyAnnotation("haha")	//value 可以省略
    public void doOther(){

    }
}

六、注解当中属性的种类

  1. 注解当中的属性可以是哪一种类型?

    属性的类型可以是: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();
    }

  2. 在使用注解给属性赋值时,数组形式用{属性值,属性值,属性值...}

  3. 在使用注解给属性赋值时,数组形式也可用{枚举值,枚举值,枚举值...}

    public class OtherAnnotationTest {
    
    <span class="hljs-comment">// 数组是大括号</span>
    <span class="hljs-meta">@OtherAnnotation(age = 25, email = {"zhangsan@123.com", "zhangsan@sohu.com"}, seasonArray = Season.WINTER)</span>
    <span class="hljs-keyword">public</span> <span class="hljs-keyword">void</span> <span class="hljs-title function_">doSome</span><span class="hljs-params">()</span>{
    
    }
    
    <span class="hljs-comment">// 如果数组中只有1个元素:大括号可以省略。</span>
    <span class="hljs-meta">@OtherAnnotation(age = 25, email = "zhangsan@123.com", seasonArray = {Season.SPRING, Season.SUMMER})</span>
    <span class="hljs-keyword">public</span> <span class="hljs-keyword">void</span> <span class="hljs-title function_">doOther</span><span class="hljs-params">()</span>{
    }
    

    }

七、通过反射获取注解对象属性的值

示例 1: 获得类上注解的属性的值

注解类:MyAnnotation

注意:如果想通过反射获取注解相关的内容。

则元注解 @Retention 的值必须是 @Retention(RetentionPolicy.RUNTIME)

否者反射获取不到,会报错!

// 只允许该注解可以标注类、方法
@Target({ElementType.TYPE, ElementType.METHOD})
// 希望这个注解可以被反射
@Retention(RetentionPolicy.RUNTIME)
public @interface MyAnnotation {
<span class="hljs-comment">/*
value属性,有默认值。
 */</span>
String <span class="hljs-title function_">value</span><span class="hljs-params">()</span> <span class="hljs-keyword">default</span> <span class="hljs-string">"北京大兴区"</span>;

}

测试类:MyAnnotationTest

@MyAnnotation("上海浦东区")
public class MyAnnotationTest {
<span class="hljs-comment">//@MyAnnotation	//编译报错</span>
<span class="hljs-type">int</span> i;

<span class="hljs-comment">//@MyAnnotation,编译报错,不能用在构造方法上</span>
<span class="hljs-keyword">public</span> <span class="hljs-title function_">MyAnnotationTest</span><span class="hljs-params">()</span>{

}

<span class="hljs-meta">@MyAnnotation</span>
<span class="hljs-keyword">public</span> <span class="hljs-keyword">void</span> <span class="hljs-title function_">doSome</span><span class="hljs-params">()</span>{

    <span class="hljs-comment">//@MyAnnotation	//编译报错</span>
    <span class="hljs-type">int</span> i;
}

}

通过反射获取注解对象属性的值:ReflectAnnotationTest

public class ReflectAnnotationTest {
    public static void main(String[] args) throws Exception{
        // 获取这个类
        Class c = Class.forName("com.example.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); // @com.bjpowernode.java.annotation5.MyAnnotation()
            // 获取注解对象的属性怎么办?和调接口没区别。
            String value = myAnnotation.value();
            System.out.println(value);
        }
    <span class="hljs-comment">// 判断String类上面是否存在这个注解</span>
    <span class="hljs-type">Class</span> <span class="hljs-variable">stringClass</span> <span class="hljs-operator">=</span> Class.forName(<span class="hljs-string">"java.lang.String"</span>);
    System.out.println(stringClass.isAnnotationPresent(MyAnnotation.class)); <span class="hljs-comment">// false</span>
}

}

示例 2:获得方法上注解上属性的值

@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.METHOD)
public @interface MyAnnotation {
    /*
    username 属性
     */
    String username();
<span class="hljs-comment">/*
password属性
 */</span>
String <span class="hljs-title function_">password</span><span class="hljs-params">()</span>;

}

public class MyAnnotationTest {

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

    }

    public static void main(String[] args) throws Exception{
        // 获取 MyAnnotationTest 的 doSome() 方法上面的注解信息。
        Class c = Class.forName("com.bjpowernode.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());	//admin
            System.out.println(myAnnotation.password());	//456456
        }
    }
}

八、案例:注解在开发中有什么用?

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

首先定义一个注解:

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

}
// 这个注解 @Id 用来标注类,被标注的类中必须有一个 int 类型的 id 属性,没有就报异常。

然后自定义一个异常类,在没有这个 id 属性显示异常信息:

// 自定义异常类
public class HasNotIdPropertyException extends RuntimeException {
    public HasNotIdPropertyException(){
}
<span class="hljs-keyword">public</span> <span class="hljs-title function_">HasNotIdPropertyException</span><span class="hljs-params">(String s)</span>{
    <span class="hljs-built_in">super</span>(s);
}

}

使用注解的类:

@MustHasIdPropertyAnnotation
public class User {
    int id;
    String name;
    String password;
}

测试类:

实现步骤:

  1. 获得使用注解的类的字节码文件:Class userClass = Class.forName("com.example.java.annotation7.User");

  2. 判断类上是否存在 Id 注解。if(userClass.isAnnotationPresent(MustHasIdPropertyAnnotation.class)){}

    1. 存在 id 注解,判断有没有 id 字段,如果有 id 字段且类型是 int 型。

      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;
          }
      }
      
    2. Id 注解或者没有 id 字段或者类型不是 int 型。

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

完整代码:

public class Test {
    public static void main(String[] args) throws Exception{
        // 获取类
        Class userClass = Class.forName("com.example.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;
                }
            }
        <span class="hljs-comment">// 判断是否合法</span>
        <span class="hljs-keyword">if</span>(!isOk){
            <span class="hljs-keyword">throw</span> <span class="hljs-keyword">new</span> <span class="hljs-title class_">HasNotIdPropertyException</span>(<span class="hljs-string">"被@MustHasIdPropertyAnnotation注解标注的类中必须要有一个int类型的id属性!"</span>);
        }

    }
}

}