对于注解,java8主要有两点改进:类型注解和重复注解
java8的类型注解扩展了注解使用的范围。在版本之前,注解只能是在声明的地方使用,现在几乎可以为任何东西添加注解:逐步变量、类与接口,就连方法的异常也能添加注解。
新增的两个注解的程序元素类型:ElementType.TYPE_USE 和 ElementType.TYPE_PARAMETER用来描述注解的新场合。ElementType.TYPE_PARAMETER表示该注解能写在类型变量的声明语句中。而ElementType.TYPE_USE 表示该注解能写在使用类型的任何语句中(例如声明语句、泛型和强制转换语句中的类型)。
对类型注解的支持,增强了通过静态分析工具发现错误的能力。原先只能在运行时发现的问题可以提前在编译的时候被排查出来。java8本身虽然没有自带类型检测的框架,但是可以通过使用Checker Framework 这样的第三方工具,自动检查和确认软件的缺陷,提高生产效率。
例如,下面的代码可以通过编译,但是运行时会报NullPointerException (空指针异常)的异常。
public class TestAnno {
public static void main(String[] args) {
Object obj = null;
obj.toString();
}
}
为了能在编译期间就自动检查出这类异常,可以通过类型注解结合Checker Framework 提前排查出来:
import org.checkerframework.checker.nullness.qual.NonNull;
public class TestAnno {
public static void main(String[] args) {
@NonNull Object obj = null;
obj.toString();
}
}
编译时自动检测结果如下:
C:\workspace\TestJava8\src\TestAnno.java:4: Warning:
(assignment.type.incompatible) $$ 2 $$ null $$ @UnknownInitialization @NonNull Object $$ ( 152, 156 )
$$ incompatible types in assignment.
@NonNull Object obj = null;
^
found : null
required: @UnknownInitialization @NonNull Object
另外,在该版本之前使用注解的第一个限制是相同的注解在同一个位置只能声明一次,不能声明多次。java8引入了重复机制,这样相同的注解可以在同一个地方声明多次。重复注解机制本身必须用@Repeatable注解。
例如,下面就是用@Repeatable 重复注解的例子:
@Retention(RetentionPolicy.RUNTIME) \\该注解存在于类文件中并在运行时可以通过反射获取
@interface Annots {
Annot[] value();
}
@Retention(RetentionPolicy.RUNTIME) \\该注解存在于类文件中并在运行时可以通过反射获取
@Repeatable(Annots.class)
@interface Annot {
String value();
}
@Annot("a1")@Annot("a2")
public class Test {
public static void main(String[] args) {
Annots annots1 = Test.class.getAnnotation(Annots.class);
System.out.println(annots1.value()[0]+","+annots1.value()[1]);
// 输出: @Annot(value=a1),@Annot(value=a2)
Annot[] annots2 = Test.class.getAnnotationsByType(Annot.class);
System.out.println(annots2[0]+","+annots2[1]);
// 输出: @Annot(value=a1),@Annot(value=a2)
}
}
注解Annot 被@Repeatable(Annots.class) 注解。Annots 只是一个容器,它包含Annot数组,编译器尽力向程序员隐藏他的存在。通过这样的方式,Test类可以被Annot注解两次。重复注解的类型可以通过getAnnotationsByType() 方法来返回。