反射机制和注解:
1. 反射机制
理解:
Reflection(反射)是被视为动态语言的关键,反射机制允许程序在执行期借助于Reflection API取得任何类的内部信息,并能直接操作任意对象的内部属性及方法。
也就是说,我们可以通过反射,去操作一个类的内部属性和方法,权限修饰符将不会起作用
当我们使用IDE编写代码时,往往输入一个方法名的一半,IDE就会帮我们自动补全代码,这其中就是用到了反射机制
1. 获取Class的实例
- 调用运行时类的属性.class
- 通过运行时类的对象的getClass()方法
- 调用Class的静态方法,forName()
- 使用类的加载器:ClassLoader
代码示例:Demo类
@Test
public void test01() throws ClassNotFoundException {
//第一种方式
Class<Person> clazz1 = Person.class;
//第二种方式
Person p = new Person();
Class<?> extends Person> clazz2 = p.getClass();
//第三种方式
Class<?> clazz3 = Class.forName("com.gec.domain.Person");
//第四种方式
Class<?> clazz4 = Demo.class.getClassLoader().loadClass("com.gec.domain.Person");
System.out.println(clazz1);
System.out.println(clazz2);
System.out.println(clazz3);
System.out.println(clazz4);
}
运行结果:
2. Class类的常用方法
Class部分代码实例:
@Test
public void test02() {
Class<Person> clazz = Person.class;
String simpleName = clazz.getSimpleName();
String name = clazz.getName();
ClassLoader classLoader = clazz.getClassLoader();
//简单类名
System.out.println(simpleName);
//全限定名称
System.out.println(name);
//类加载器
System.out.println(classLoader);
}
3. Constructor
理解:
Constructor类提供了一个类的单个构造函数的信息和访问权限
Constructor类代码示例:
@Test
public void test03() {
Class<Person> clazz = Person.class;
//获取由所有已声明构造方法组成的数组
Constructor[] dcs = clazz.getDeclaredConstructors();
for (Constructor<Person> constructor : dcs) {
System.out.println(constructor);
}
}
@Test
public void test04() throws Exception {
Class<Person> clazz = Person.class;
//获取单个构造方法,参数为构造方法的形参
Constructor<Person> dc = clazz.getDeclaredConstructor(String.class);
//因为该构造方法是私有的,需要设置访问权限
dc.setAccessible(true);
Object obj = dc.newInstance("TC");
System.out.println(obj);
}
运行结果:
4. Field
Field提供有关类或接口的单个变量的信息和动态访问。 反射的变量可以是类(静态)变量或实例变量
Field相关方法代码示例:
@Test
public void test05() throws Exception {
Class<Person> clazz = Person.class;
//获取所有已声明属性的数组
Field[] fields = clazz.getDeclaredFields();
for (Field field : fields) {
System.out.println(field);
}
}
@Test
public void test06() throws Exception {
Class<Person> clazz = Person.class;
//获取单个已声明属性
Constructor<Person> dc = clazz.getDeclaredConstructor(String.class, Integer.class);
Person p = dc.newInstance("TC", 20);
System.out.println(p);
Field name = clazz.getDeclaredField("name");
//设置访问权限
name.setAccessible(true);
name.set(p, "zzk");
System.out.println(p);
}
运行结果:
5. Method
方法在类或接口上提供有关单一方法的信息和访问权限。 反射的方法可以是类方法或实例方法(包括抽象方法)
Method类相关方法代码示例:
@Test
public void test08() throws Exception {
Class<Person> p = Person.class;
//获取所有已经声明方法的数组
Method[] methods = p.getDeclaredMethods();
for (Method method : methods) {
System.out.println(method);
}
}
@Test
public void test09() throws Exception {
Class<Person> clazz = Person.class;
//获取单个已声明方法
Constructor<Person> dc = clazz.getDeclaredConstructor(String.class, Integer.class);
//设置访问权限
dc.setAccessible(true);
Person p = dc.newInstance("TC", 20);
//注意参数类型也要与方法的形参类型相同
Method method = clazz.getDeclaredMethod("sum", Integer.class,Integer.class);
Integer result = (Integer)method.invoke(p, 20,10);
System.out.println(result);
System.out.println(p);
}
运行结果:
值得注意的是,在利用反射机制获取类的构造器和方法时
参数类型也要与方法的形参类型相同
以上三个类都有个共同的方法以访问私有成员,setAccessible(boolean flag),参数为true时即可操作私有成员,这种操作被称为暴力反射
2. 注解
理解:
注解(Annotation) 其实就是代码里的特殊标记, 这些标记可以在编译, 类加载, 运行时被读取, 并执行相应的处理。通过使用 Annotation,程序员可以在不改变原逻辑的情况下, 在源文件中嵌入一些补充信息。
元注解:
定义注解的注解
定义一个注解类:
Retention:生命周期以及改注解的有效范围
RetentionPolicy.RUNTIME : 运行时
RetentionPolicy.SOURCE : 源文件
RetentionPolicy.CLASS : 类文件
声明的位置,类上,属性上,方法上
一般使用:
ElementType.TYPE : 类
ElementType.FIELD : 属性
ElementType.METHOD : 方法
自定义注解类:
//
@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
public @interface Table {
//注解的属性定义方法如下
String value();
//可以包括默认值
String name default "TC";
}
Annotation相关的方法与前面讲的三个同理,需要时翻阅API文档即可理解