使用Annotation的例子
学习《疯狂java讲义》
使用标记Annotation
- 首先,我们先定义一个不含任何成员变量的Annotation类,也就是标记Annotation。
//Retention注释用于指定Testable注释可以保留多久,这里是保留在运行期间
//Target注释用于指定Testable注释能修饰的目标,这里指定了只能是方法
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.METHOD)
//定义Annotation用@interface关键字。
//我们还记得定义接口的时候用的是interface关键字
public @interface Testable{}
- 然后我们再定义一个类,里面定义一些方法,有的用了Testable注释,有的没有用
public class MyTest {
@Testable
public static void m1(){}
public static void m2(){}
@Testable
public static void m3(){
throw new RuntimeException("Boom");
}
public static void m4(){}
@Testable
public static void m5(){}
public static void m6(){}
@Testable
public static void m7(){
throw new RuntimeException("Crash");
}
public static void m8(){}
}
- 有了这些注释,并不能真正的影响程序的运行。因此我们还需要为这些注释提供一个注释处理工具。它会分析目标类,如果目标类方法使用了@Testable注释,则通过反射来运行该测试方法。
public class TestProcessor{
public static void process(String clazz) throws ClassNotFoundException {
int passed = 0;
int failed = 0;
for(Method m : Class.forName(clazz).getMethods()){
if (m.isAnnotationPresent(Testable.class)){
try{
m.invoke(null);
passed++;
} catch (Exception ex){
System.out.printf("方法"+m+"异常"+ex.getCause()+"\n");
failed++;
}
}
}
System.out.println("共运行了"+(passed+failed)+"方法,"+"成功了"+passed+",失败了"+failed);
}
}
- 定义了注释,有了目标类,又定义了注释工具类,现在可以再加个主方法,然后把上面的都运行起来啦
public class RunTest {
public static void main(String[] args)throws Exception{
TestProcessor.process("MyTest");
}
}
-
Ok,现在看看运行的结果
-
遇到的问题:
在上面的RunTest类中,使用TestProcessor的process方法时,我们直接使用了“MyTest”。但是,在实际中,我总会报这样的错误:
问题出在哪了呢?
我们看到,异常主要在main线程中,发生了ClassNotFoundException,也就是说没有找到"MyTest"这个类。咋回事呢???这就得说回我的目录结构了。在这个工程中,上面定义的这几个类都是放在一个包里的不同文件,并且在每个文件中都声明了package,如下:
所以呢,在这里,我不直接用“MyTest”,而是把包名加上,也就是TestProcessor.process("ChapterAnnotation.MyTest");
,然后,BOOM!一切都正常运行了!也就是上面正常运行的结果。