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})- 表示该注解可以出现在:
- 构造方法上
- 字段上
- 局部变量上
- 方法上
- 类上…
- …
- @Target(ElementType.METHOD):表示“被标注的注解”只能出现在方法上。
关于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 {@MyAnnotationprivate int no;@MyAnnotationpublic AnnotationTest01(){}@MyAnnotationpublic static void m1(){@MyAnnotationint i = 100;}@MyAnnotationpublic void m2(@MyAnnotationString name,@MyAnnotationint 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 {//@Overrideprivate int no;@Overridepublic String toString() {return "toString";}
}
示例代码03:
// 表示这个类已过时。
@Deprecated
public class AnnotationTest03 {@Deprecatedprivate int no;@Deprecatedpublic void doSome() {System.out.println("doSome....");}// Deprecated这个注解标注的元素已过时。// 这个注解主要是向其它程序员传达一个信息,告知已过时,有更好的解决方案存在。@Deprecatedpublic 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();}
运行结果:
示例代码04:
public @interface MyAnnotation {/*** 我们通常在注解当中可以定义属性,以下这个是MyAnnotation的name属性。* 看着像1个方法,但实际上我们称之为属性name。* @return*/String name();/*颜色属性*/String color();/*年龄属性*/int age() default 25;//属性指定默认值}
public class MyAnnotationTest {// 报错的原因:如果一个注解当中有属性,那么必须给属性赋值。(除非该属性使用default指定了默认值。)/*@MyAnnotationpublic 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 {// 报错原因:没有指定属性的值。/*@MyAnnotationpublic 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);}
}
运行结果:
示例代码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());}}
}
运行结果:
需求:
- 假设有这样一个注解,叫做:@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类型的idisOk = 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 {}
运行结果:
Java基础进阶注解相关推荐
- 第二十八节:Java基础-进阶继承,抽象类,接口
前言 Java基础 - 进阶继承,抽象类,接口 进阶继承 class Stu {int age = 1; } class Stuo extends Stu {int agee = 2; } class ...
- 第一阶段>>>Java基础进阶 OOP/Reflect/注解/IO流/API核心
目录 >>>java基础01-结构以及类型 >>>java基础02-初识数组 >>>java基础03-面向对象OOP >>>ja ...
- 分享 java 基础 + 进阶精简资料(视频 + 源码 + 就业项目 + 面试报装)
前言 有些同学可能会觉得奇怪,为什么会写这么一篇文章,Java 基础不是大学就学过了吗,就这呀,这有什么好写的? 但是从事开发越久才越知道基础有多重要,不止 Java 基础,还有计算机网络基础与操作系 ...
- JAVA基础,注解反射机制
文章目录 注解 (非常重要) 什么是注解 基本注解 @Override @Deprecated @SuppressWarnings @SafeVarargs @FunctionalInterface ...
- 《Java基础+进阶》-浙江大学MOOC
基础篇: 1.温度转换 题目内容: 写一个将华氏温度转换成摄氏温度的程序,转换的公式是: °F = (9/5)*°C + 32 其中C表示摄氏温度,F表示华氏温度. 程序的输入是一个整数,表示华氏温度 ...
- 自学JAVA基础 进阶版
字符串 1.API 1.1 API概述 API(Application Programming Interface):应用程序编程接口 Java API:指的就是JDK中提供的各种功能的Java类.这 ...
- java基础进阶一:String源码和String常量池
作者:NiceCui 本文谢绝转载,如需转载需征得作者本人同意,谢谢. 本文链接:http://www.cnblogs.com/NiceCui/p/8046564.html 邮箱:moyi@moyib ...
- 【编程语言】Java基础进阶——面向对象部分
断断续续终于把男神的Java都看完了,只是囫囵吞枣走马观花罢了,还有好多地方不是很理解,很多细节都没能理清.之后再边动手边复习一遍,然后开始尝试做点东西吧! 0. 一些小tips ·为避免出现错误提示 ...
- JAVA基础进阶day01
最近一段时间的自学重心是安卓底层.首先啃一下java.新年伊始,向着 知识的海洋急行军,世界人民团结万岁-. 笔记方式为代码加注释的方式: 一.最基础 public class Hello {publ ...
最新文章
- PE文件结构详解(六)重定位
- javaScript转换日期合格式
- PMP读书笔记(第2章)
- OSPF高级特性——LSA-3(Network Summary LSA)的过滤、路由协议——管理距离的修改、外部路由的metric值的修改
- C#LeetCode刷题之#217-存在重复元素(Contains Duplicate)
- 《季羡林先生》读书笔记-3
- jQuery二维码插件 jquery.qrcode.js
- 《Xcode实战开发》——1.1节下载
- 思科路由器的介绍及实战
- 数据分析——人力资源
- 电子工程可以报考二建_我是电子信息工程专业,可以报考二级建造师吗?要..._二建考试_帮考网...
- Windows 11和安全启动
- [ESP][驱动]ST7701S RGB屏幕驱动
- 什么是DNS?DNS的作用?
- 深度学习原理与框架-RNN网络框架-LSTM框架 1.控制门单元 2.遗忘门单元 3.记忆门单元 4.控制门单元更新 5.输出门单元 6.LSTM网络结构...
- linux cache控制 内核,Linux内核中cache的实现
- 复指数与高斯函数乘积的傅里叶变换_测量波函数
- Java关键字-static
- 2021全球程序员收入报告发布,字节跳动成为中国唯一上榜的公司
- Win7下缩小任务栏宽度