1. 要求
实现自定义标签@MyTag,把其作为测试方法的标识。
例用反射知识,运行被@MyTag修饰的无参方法。
2. 考查
如何自定义标签?
如何使用反射?
3. 代码示例
3.1 标签类: MyTag.java
(1)@Retention:作用是定义被它所注解的注解保留多久。
一共有三种策略,定义在RetentionPolicy枚举中。
这3个生命周期分别对应于:Java源文件(.java文件) ---> .class文件 ---> 内存中的字节码
source:注解只保留在源文件,当Java文件编译成class文件时,注解被遗弃;被编译器忽略
class:注解被保留到class文件,但jvm加载class文件时被遗弃,这是默认的生命周期
runtime:注解不仅被保存到class文件中,jvm加载class文件之后,仍然存在
(2)@Target用来表示注解作用范围,超过这个作用范围,编译的时候就会报错。
@Target(ElementType.TYPE):接口、类、枚举、注解
@Target(ElementType.FIELD):字段、枚举的常量
@Target(ElementType.METHOD):方法
@Target(ElementType.PARAMETER):方法参数
@Target(ElementType.CONSTRUCTOR) :构造函数
@Target(ElementType.LOCAL_VARIABLE):局部变量
@Target(ElementType.ANNOTATION_TYPE):注解
@Target(ElementType.PACKAGE):包
(3)@interface Annotation{ } :自定义一个注解 @Annotation,一个注解是一个类。
@interface自定义注解,自动继承了java.lang.annotation.Annotation接口,由编译程序自动完成其他细节。
在定义注解时,不能继承其他的注解或接口。
使用@interface来声明一个注解。
package MyTagDemo;
import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.METHOD)
public @interface MyTag {
}
3.2 测试类:TestTag.java
package MyTagDemo;
public class TestTag {
@MyTag
public void login(){
System.out.println("login");
}
@MyTag
public void test1(String p){
System.out.println("test1");
}
@MyTag
public void test2(){
System.out.println("test2");
}
@MyTag
public void logout(){
System.out.println("logout");
}
}
3.3 入口类:MyDemo.java
package MyTagDemo;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
public class MyDemo {
public static void main(String[] args) throws ClassNotFoundException, NoSuchMethodException, IllegalAccessException, InvocationTargetException, InstantiationException {
Class<?> clazz = Class.forName("MyTagDemo.TestTag");
Object obj = clazz.getDeclaredConstructor().newInstance(); //创建实例
Method[] methods = clazz.getDeclaredMethods(); //获取所有声明的方法
for(int i=0; i<methods.length; i++){
//判断方法是否带有@MyTag标签,并且无参。是,则invoke运行。
//方法名.getAnnotation():可获取该方法的注解
//方法名.getParameterCount():可获取该方法的参数个数
if(methods[i].getAnnotation(MyTag.class) != null && methods[i].getParameterCount() == 0){
System.out.println("测试用例名称是:" + methods[i].getName());
methods[i].invoke(obj); //传实例对象,因为只处理无参方法所以这里不用传参数
}
}
}
}
输出结果:
测试用例名称是:login
login
测试用例名称是:logout
logout
测试用例名称是:test2
test2