项目的目录结构如下
package _zjTest;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
/*
* 1、RetentionPolicy.SOURCE:注解只保留在源文件,当Java文件编译成class文件的时候,注解被遗弃;
2、RetentionPolicy.CLASS:注解被保留到class文件,但jvm加载class文件时候被遗弃,这是默认的生命周期;
3、RetentionPolicy.RUNTIME:注解不仅被保存到class文件中,jvm加载class文件之后,仍然存在;
这3个生命周期分别对应于:Java源文件(.java文件) ---> .class文件 ---> 内存中的字节码。
那怎么来选择合适的注解生命周期呢?
首先要明确生命周期长度source < class < runtime ,所以前者能作用的地方后者一定也能作用。
*/
@Retention(RetentionPolicy.RUNTIME) // 保留策略
public @interface MyAnnotation {
//注解
}
package _zjTest;
public class MyUser {
@MyAnnotation
public void say1() {
System.out.println("执行1");
}
public void say2() {
System.out.println("执行2");
}
}
package _zjTest;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
public class Test {
/* 一、newInstance()和new()区别:
1、两者创建对象的方式不同,前者是实用类的加载机制,后者则是直接创建一个类:
2、newInstance创建类是这个类必须已经加载过且已经连接,new创建类是则不需要这个类加载过
3、newInstance: 弱类型(GC是回收对象的限制条件很低,容易被回收)、低效率、只能调用无参构造,new 强类型
*/
public static void main(String[] args) {
// TODO Auto-generated method stub
// Java的反射机制
//MyUser.class-----_zjTest.MyUser
Class cl = MyUser.class;
//cl.getMethods()-----java.lang.reflect.Method-----@7852e922
Method[] mds = cl.getMethods();
for (Method md : mds) {
// 判断某方法是否被指定注解标注
boolean f = md.isAnnotationPresent(MyAnnotation.class);
if (f == true) {
try {
// 让md这个方法执行--invoke援引
//invoke(调用)就是调用Method类代表的方法。可以实现动态调用,例如可以动态的传人参数,可以把方法参数化。
md.invoke(cl.newInstance(), null);
} catch (IllegalAccessException | IllegalArgumentException | InvocationTargetException
| InstantiationException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
}
}
}
执行结果如下
不带注解的方法没有执行,这里的Test类是相当于一个注解解析类