注解,或者叫做注释类型,英文单词是: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 {
@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();
}
运行结果:
示例代码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);
}
}
运行结果:
示例代码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类型的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 {
}
运行结果:
版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 举报,一经查实,本站将立刻删除。
文章由极客之音整理,本文链接:https://www.bmabk.com/index.php/post/94236.html