一、理解Annotation
- ① jdk5.0新增的功能
- ② Annotation 其实就是代码里的特殊标记,这些标记可以在编译,类加载,运行时被读取,并执行相应的处理
- 通过使用Annotation,程序员可以在不改变代码原有的逻辑下,在源代码嵌入一些补充信息
- ③ 在javaSE中,注解的使用目的比较简单,例如标记过时的功能,忽略警告等等。
- 在JavaEE中注解占据了更重要的角色,例如用来配置应用程序,代替JavaEE旧版本锁遗留的繁冗代码和xml配置等
- ④ 未来的开发模式都是基于注解的JPA是基于注解的Spring2.5以上都是基于注解的,注解是一种趋势,
- 一定程度上可以说:框架 = 注解 + 反射 + 设计模式
二、Annotation的使用实例
- ① 生成文档的相关注解
- ② 在编译是进行格式检查(JDK内置的三个基本注解)
- @Override: 限定重写父类方法。该注解只能用于方法
- @Deprecated: 用于表示所修饰的元素(类, 方法 , 构造器等等)已过时。通常是因为所修饰的结构危险或存在更好的选择,过时是可以用的,意义只是作为一种提示,因为原来的项目中用的老的代码必须要能用,过时是给我们后面做开发的提示
- @SuppressWarnings: 抑制编译器警告
- ③ 跟踪代码的依赖性,实现替代配置文件功能
三、如何自定义注解:参照@SuppressWarnings的定义
- ① 定义新的Annotation类型使用@inferface关键字
- ② 自定义注解自动继承java.lang.annotation.Annotation接口
- ③ 可以在定义Annotation的成员变量时为其指定初始值,指定成员变量的初始值可以使用default关键字
- ④ 如果定义的注解中含有配置参数,那么使用时必须指定参数值,除非它有默认值。
- 定义成员内部成员,一般使用value表示
- 格式为“ 参数名 = 参数值”,如果智原一个参数成员,且名称为value 可以省略“value = ”
- ⑤ 没有成员定义的 Annotation 就称为标记,包含成员变量的Annotation称为元数据Annotation
- 元数据:对现有数据的一个修饰 String name = “数据”
- 注意:自定义注解必须配上注解的信息处理流程才有意义
- 自定义注解通常会指明两个元注解:Retention、Target
public class Annotation {
public static void main(String[] args) {
Person p = new Student();
p.walk();
}
}
//jdk8之前的写法
//@MyAnnotations({@MyAnnotation(value = "hi"),@MyAnnotation(value = "hi")})
//重复注解
@MyAnnotation(value = "hi")
@MyAnnotation(value = "hell")
class Person {
private String name;
private int age;
public Person(String name, int age) {
this.name = name;
this.age = age;
}
public Person() {
}
@SuppressWarnings("")
public void walk() {
System.out.println("人走路");
}
public void eat() {
System.out.println("人吃饭");
}
}
interface Info1 {
void show();
}
class Student extends Person implements Info1 {
@Override
public void walk() {
System.out.println("学生走路");
}
@Override
public void show() {
System.out.println("显示");
}
}
四、jdk提供的 四种 元注解
-
元注解:对现有的注解进行解释说明的注解
-
① Retention:指定所修饰的Annotation的生命周期
-
作用:表示需要在什么级别保存该注释信息,用于描述注解的生命周期(即:被描述的注解在什么范围内有效)
-
取值(RetentionPoicy)有:是一个枚举类
-
1.SOURCE:在源文件中有效(即源文件保留)
-
2.CLASS:在class文件中有效(即class保留)
-
3.RUNTIME:在运行时有效(即运行时保留):只有声明为RUNTIME的注解,才能通过反射获取
-
② Target:用于指定被修饰的Annotation 能用于修饰哪些程序元素
-
比如:类、接口、成员变量、修饰方法,构造器等等
-
以下的两种出现频率较低
-
- ③ Documented:生成API帮助文档时显示注解。表示所修饰的注解在被javadoc解析时,保留下来。
-
- ④ Inherited:被他修饰的Annotation将具有继承性,其子类也自动具该注解
-
5.如何获取注解信息:通过反射来获取、调用
-
前提:要求此注解的元注解中声明的(Retention)声明周期试试:RUNTIME
@Inherited
@Repeatable(MyAnnotations.class)
@Target({TYPE, FIELD, METHOD, PARAMETER, CONSTRUCTOR, LOCAL_VARIABLE,TYPE_PARAMETER,TYPE_USE})
@Retention(RetentionPolicy.RUNTIME)
public @interface MyAnnotation {
//有默认值时,可以不在使用时赋值
String value() default "hello";
}
//缺少一个编译不通过,就是编译通过,运行也会报错
@Inherited
@Target({TYPE, FIELD, METHOD, PARAMETER, CONSTRUCTOR, LOCAL_VARIABLE})
@Retention(RetentionPolicy.RUNTIME)
public @interface MyAnnotations {
MyAnnotation[] value();
}
六、jdk8 这个注解的新特性:可重复注解,类型注解
6.1可重复注解 ① 在MyAnnotation上面声明@Repeatable,成员值为MyAnnotations.class
- ② MyAnnotation的Target和Retention与MyAnnotations的相同
6.2 类型注解:
- ElementType.TYPE_PARAMETER 表示该注解能写在类型的变量的声明语句中(如:泛型声明)
- ElementType.TYPE_USE 表示该注解能写在使用类型的任何语句中。
//类型注解,在MyAnnotation的@Target中声明TYPE_PARAMETER
class Generic<@MyAnnotation T>{
//在MyAnnotation的@Target中声明TYPE_USE,表示只有是类型的地方都可以使用
public void show() throws @MyAnnotation RuntimeException{
ArrayList<@MyAnnotation String> list = new ArrayList<>();
}
}
面试题
比较 throw 和 throws 的异同
- 两者没有多大的关系
- throw:生成一个异常对象,一般是在方法内部 <-> 自动抛出异常对象
- throws:处理异常的方式,使用在声明处的末尾,表示向调用该类的位置抛出异常,不在该类解决
谈谈你对同步代码块中的同步监视器和共享数据的理解和各自的要求
- 同步监视器:锁,①任何一个对象都可以充当 ②多个线程共用一个锁
- 共享数据:多个线程共同操作的数据,需要使用同步机制将操作共享数据的代码包起来