注解:
概念:注解,也叫元数据,一种代码级别的说明。它是jdk1.5及以后版本引入的一个特性,与类,接口,枚举是同一个层次。它可以声明在包,类,字段,方法,局部变量,方法参数等的前面,用来对这些元素进行说明,注解。
概念描述:jdk1.5之后的新特性
用来说明程度的
使用注解:@注解名称
作用分类:
1 编写文档:通过代码里标识的注解生成文档【生成文档doc文档】
2 代码分析:通过代码里表识的注解对代码进行分析【使用反射】
3 编译检查:通过代码里标识的注解让编译器能够实现基本的编译检查[override]
jdk中预定义的一些注解:
@Override:检测被该注解标注的方法是否继承自父类(接口)的
@Oeprecated:该注解标注的内容,表示已经过时
@SuppressWarnings:压制警告。
例子:
@SuppressWarning(“all”)//压制警告,作用:使得该类中所有的警告信心不在显示
public class AnnoDemo{
@Overrider//继承自父类的toString()方法
public String toString(){
rerturn super.toString
}
@Deprecated //表示该方法已经过时
public void show1(){
//该方法已被替换
}
}
自定义注解:
* 格式:
* 元注解
* public @interface 注解名称{} 反编译:javap命令
* : * 本质:注解本质上就是一个接口,该接口默认继承Annotation接口
* public interface MyAnno extends java.lang.annotation.Annotation{ }
* 属性:接口中可以定义的抽象方法
* 要求:
* 1 属性的返回值类型
* 基本数据类型
* String
* 枚举
* 注解
* 以上类型的数组
* 例如:
* public @interface MyAnno{
int show1();
Sting show2();
Person per();//枚举类型
MyAnno2 anno2();//注解类型
String [] strs();
int[] a();
定义了属性,在使用时需要给属性赋值:
例如:
public @interface MyAnno{
int age();
String name();
//String name() default "张三" 默认值,在使用时这个属性不赋值默认为“张三”
}
使用:
@MyAnno(age=12,name="zhangsan’)
1 如果定义属性时,使用default关键字给属性默认初始化值,则使用注解时,可以不进行属性的赋值。
2 如果只有一个属性需要赋值,并且属性的名称是value,则value可以省略,直接定义值即可。
3 数组赋值时,值使用{}包裹。如果数组中只有一个值,则{}可以省略。
元注解:用于描述注解的注解
* @Target:描述注解能够作用的位置
@Retention:描述注解被保留的阶段
@Documented:描述注解是否被抽取到api文档中
@Inherited:描述注解是否被子类继承。
1 @Target(value={ElementType.TYPE})//表示该MyAnno3注解只能作用于类上。
value的值是一个数组。
ElementType取值:
TYPE:可以作用于类上
METHOD:可以作用于方法上
FIELD:可以作用于成员变量上。
@Retention:描述注解被保留的阶段
@Retention(RetentionPolicy.RUNTIME):当前被描述的注解,会保留到class字节码文件中,并被JVM读取到。
在程序中使用(注解):在程序中获取注解的属性值,注解大多数是替换配置文件的。
例子:
创建一个框架类:
public class ReflectTest{
public static void main(String[] args) throws Exception{
// 1解析注解
// 获取该类的字节码文件对象
获取注解定义的位置对象 (Class,Method,Field)
Class< ReflectTest> reflectTestClass=ReflectTest.class;
// 获取上边的注解对象
Pro an=reflectTestClass.getAnnotation(Pro.class);//其实就是在内存中生成了一个该类注解接口的子类实现对象
// 调用注解对象中定义的抽象方法,获取返回值
String className=an.className();
String methodName=an.methodName();
//使用反射执行方法:
//加载目标类进入类存
Class cls=Class.forName(className);
//创建对象
Object obj=cls.newInstance();
//获取方法对象
Method method=cls.getMethod(methodName);
//执行方法
method.invoke(obj);
}
}
实例:
需求:
自定义手写一个注解框架,实现检测方法有没有问题,如果有问题,就不执行该方法,把错误生成日志,写到文件中。
1 定义注解接口:
public @interface Check{
}