注解
Java编程思想中这样定义注解:注解被称为元数据,为我们在代码中添加信息提供了一种形式化的方法。使我们在稍后某个时刻非常方便的使用这些数据。
但元数据是啥意思目前还不是很理解。。。。元数据???
注解仅仅是一种元数据,和业务逻辑没有任何关系,所以注解定义中没有任何逻辑处理。
Java中有四种注解专门负责新注解的创建,稍后会学习。
Test.java:一个标记注解类
- 注解的定义和接口的定义非常相似,事实上,与其他任何Java接口一样
- 注解也会编译成class文件。注解和接口最明显的区别是在@符号上,即注解定义时为@interface,接口为interface
- 定义注解时,需要一些元注解的修饰。如@Target、@Retention
- 一般注解都会包含某些元素以表示某些值,注解的元素看起来就像接口的方法。唯一的区别就是可以为其制定默认值。
- 没有元素的注解叫做标记注解,就像本例中的Test注解
- @Target用来定义你将注解用在什么地方,可以在类、方法、变量、参数、包中使用
- @Rectetion用来定义该注解在哪一个级别可用(源代码SOURCE、类文件CLASS、或者运行时RUNTIME)
@Target(ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)
public @interface Test {
}
UseCase.java:一个简单的注解用例。
@Target(ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)
public @interface UseCase {
public int id();
public String description() default "no description";
}
上例中id和description类似方法定义,称为注解元素。编译器会对id进行类型检查,如果在注解中某个方法没有给出description的值,则该注解会使用默认值。
用例:PassWordUtils.java
public class PassWordUtils {
@UseCase(id = 47,description = "Passwords must contain number")
public boolean validatePassword(String password){
return (password.matches("\\w*\\d\\w*"));
}
@UseCase(id = 48)
public String encryptPassword(String password){
return new StringBuilder(password).reverse().toString();
}
@UseCase(id = 49,description = "New password can not equal previoisly")
public boolean checkForNewPassword(List<String> prePassword,String password){
return prePassword.contains(password);
}
}
可以看到注解中的id和description会出现在方法中注解声明后的括号里,编译时会检查。
如未在注解中定义默认值,则在使用注解时必须显示给出值,否则会报错
元注解
Java目前内置的四种元注解。元注解专门负责注解其他的注解
- Target:表示该注解能用在什么地方。可能的ElementType参数包括:
- CONSTRUCTOR:构造器声明
- FIELD:域声明
- LOCAL_VARIABLE:局部变量声明
- METHOD:方法声明
- PACKAGE:包声明
- PARAMETER:参数声明
- TYPE:类、接口或enum声明
- Retention:表示在什么级别保存该信息。可选的RetentionPolicy包括
- SOURCE:注解将被编译器丢弃
- CLASS:注解在class文件中可用,但会被VM丢弃。
- RUNTIME:VM将在运行期间也保留注解,因此可以通过反射机制读取注解的信息。
- Documented:将此注解保存在javadoc中
- Inherited:允许子类继承父类中的注解。
编写注解处理器
注解处理器就是用来读取注解的工具。Java使用反射机制来构造注解处理器。
注解处理器:UseCaseTracker.java
public class UseCaseTracker {
public static void tracUsecases(List<Integer> useCases,Class<?> cl){
//通过反射,得到相应类所声明的方法
for(Method m: cl.getDeclaredMethods()){
UseCase uc = m.getAnnotation(UseCase.class);
if(uc != null){
System.out.println("Found Use Case " + uc.id()
+ "" + uc.description());
useCases.remove(new Integer(uc.id()));
}
}
for(int i : useCases){
System.out.println("Waring: Missing use case-" + i);
}
}
public static void main(String[] args) {
List<Integer> useCases = new ArrayList<>();
Collections.addAll(useCases, 47,48,49,50);
tracUsecases(useCases, PassWordUtils.class);
}
}
输出:
Found Use Case 49 New password can not equal previoisly
Found Use Case 47 Passwords must contain number
Found Use Case 48 no description
Waring: Missing use case-50
该注解处理器用到了两个反射方法getDeclaredMethod()和getAnnotation()。他们都属于AnnotatedElement接口(Class、Method、Field)都实现了该接口。getAnnotation()方法返回是定类的注解对象,在这里就是UseCase,在通过注解类定义的元素方法得到相应的元素值。
注解元素
注解元素可用的类型
- 所有基本类型(int、float、boolean等)
- String
- Class
- enum
- Annotation
- 以上所有类型的数组
注解元素必须有确认的值,或者提供默认值,或者在使用时赋值,总之不允许不赋值。
注解不能被继承
Annotation-Processing:注解处理器
注解处理器不是在运行时通过反射机制运行处理的注解,而是在编译时处理的注解。
一个特定注解的处理器以** java 源代码(或者已编译的字节码)作为输入,然后以一些文件(通常是.java文件)作为输出。那意味着什么呢?你可以生成 java 代码!这些 java 代码在生成的.java文件中。因此你不能改变已经存在的java类,例如添加一个方法**。这些生成的 java 文件跟其他手动编写的 java 源代码一样,将会被 javac 编译。
AbstractProcessor
让我们来看一下处理器的 API,所有的处理器都继承了AbstractProcessor。如下:
public class MyPocessor extends AbstractProcessor{
@Override
public boolean process(Set<? extends TypeElement> annotations, RoundEnvironment roundEnv) {
return false;
}
@Override
public Set<String> getSupportedAnnotationTypes() {
return super.getSupportedAnnotationTypes();
}
@Override
public SourceVersion getSupportedSourceVersion() {
return super.getSupportedSourceVersion();
}
@Override
public synchronized void init(ProcessingEnvironment processingEnv) {
super.init(processingEnv);
}
}
init(ProcessingEnvironment processingEnv):所有的注解处理器类都必须有一个无参构造函数。然而,有一个特殊的方法init(),它会被注解处理工具调用,以ProcessingEnvironment作为参数。ProcessingEnvironment 提供了一些实用的工具类Elements, Types和Filer我们在后面将会使用到它们。
process(Set<? extends TypeElement> annoations, RoundEnvironment env):这类似于每个处理器的main()方法。你可以在这个方法里面编码实现扫描,处理注解,生成 java 文件。使用RoundEnvironment参数,你可以查询被特定注解标注的元素(原文:you can query for elements annotated with a certain annotation )。后面我们将会看到详细内容。
getSupportedAnnotationTypes():在这个方法里面你必须指定哪些注解应该被注解处理器注册。注意,它的返回值是一个String集合,包含了你的注解处理器想要处理的注解类型的全称。换句话说,你在这里定义你的注解处理器要处理哪些注解。
getSupportedSourceVersion(): 用来指定你使用的 java 版本。通常你应该返回SourceVersion.latestSupported()不过,如果你有足够的理由坚持用 java 6 的话,你也可以返回SourceVersion.RELEASE_6 我建议使用SourceVersion.latestSupported()在 Java 7 中,你也可以使用注解的方式来替代重写