注解的语法
元注解:
1.@Target,
2.@Retention,
3.@Documented,
4.@Inherited
@Target说明了Annotation所修饰的对象范围:Annotation可被用于 packages、types(类、接口、枚举、Annotation类型)、类型成员(方法、构造方法、成员变量、枚举值)、方法参数和本地变量(如循环变量、catch参数)。在Annotation类型的声明中使用了target可更加明晰其修饰的目标。
取值(ElementType)有:
1.CONSTRUCTOR:用于描述构造器
2.FIELD:用于描述域
3.LOCAL_VARIABLE:用于描述局部变量
4.METHOD:用于描述方法
5.PACKAGE:用于描述包
6.PARAMETER:用于描述参数
7.TYPE:用于描述类、接口(包括注解类型) 或enum声明
@Retention:表示需要在什么级别保存该注释信息,用于描述注解的生命周期(即:被描述的注解在什么范围内有效)
取值(RetentionPoicy)有:
1.SOURCE:在源文件中有效(即源文件保留)
2.CLASS:在class文件中有效(即class保留)
3.RUNTIME:在运行时有效(即运行时保留)
@Documented:用于描述其它类型的annotation应该被作为被标注的程序成员的公共API,因此可以被例如javadoc此类的工具文档化。Documented是一个标记注解,没有成员。
@Inherited 元注解是一个标记注解,@Inherited阐述了某个被标注的类型是被继承的。如果一个使用了@Inherited修饰的annotation类型被用于一个class,则这个annotation将被用于该class的子类。
@Repeatable
Repeatable Java SE 8中引入的,@Repeatable注释表明标记的注释可以多次应用于相同的声明或类型使用(即可以重复在同一个类、方法、属性等上使用)。
@Deprecated标志的类,都是不鼓励使用的类,如果使用或者进行重写,程序会发出警告。
自定义注解
@interface用来声明一个注解,其中的每一个方法实际上是声明了一个配置参数。方法的名称就是参数的名称,返回值类型就是参数的类型(返回值类型只能是基本类型、Class、String、enum)。可以通过default来声明参数的默认值。
定义注解格式:
public @interface 注解名 {定义体}
注解参数的可支持数据类型:
1.所有基本数据类型(int,float,boolean,byte,double,char,long,short)
2.String类型
3.Class类型
4.enum类型
5.Annotation类型
6.以上所有类型的数组
Annotation类型里面的参数该怎么设定:
第一,只能用public或默认(default)这两个访问权修饰.例如,String value();这里把方法设为defaul默认类型;
第二,参数成员只能用基本类型byte,short,char,int,long,float,double,boolean八种基本数据类型和 String,Enum,Class,annotations等数据类型,以及这一些类型的数组.例如,String value();这里的参数成员就为String;
第三,如果只有一个参数成员,最好把参数名称设为"value",后加小括号.例:下面的例子FruitName注解就只有一个参数成员。
自定义注解实例
import java.lang.annotation.Annotation;
import java.lang.reflect.Field;
import java.util.Arrays;
public class TestDemo {
public static void main(String[] args) throws NoSuchFieldException, IllegalAccessException {
Student student = new Student();
//取得成员变量的值
Field[] fields = Student.class.getDeclaredFields();
for (Field field : fields) {
Annotation[] declaredAnnotations = field.getDeclaredAnnotations();
System.out.println(Arrays.asList(field));
System.out.println(Arrays.asList(declaredAnnotations));
}
//取得成员变量的值
Field field2 = Student.class.getDeclaredField("name");
//打开权限
field2.setAccessible(true);
//判断属性是否有注解
if (field2.isAnnotationPresent(FieldName.class)) {
//获取属性上的注解值
FieldName test = field2.getAnnotation(FieldName.class);
String name = test.value();
System.out.println(name);
//赋值
field2.set(student, name);
}
Field field1 = Student.class.getDeclaredField("age");
field1.setAccessible(true);
if (field1.isAnnotationPresent(FieldName.class)) {
FieldName test2 = field1.getAnnotation(FieldName.class);
int age = Integer.valueOf(test2.value());
field1.set(student, age);
}
System.out.println(student);
}
}