反射

反射机制允许程序在执行期借助于ReflectionAPI取得任何类的内部信息,并能操作对象的属性及方法。反射在设计模式和框架底层都会用到。

加载完类之后,在堆中就产生了一个Class类型的对象,这个对象包含了类的完整结构信息。通过这个对象得到类的结构。这个对象就像一面镜子,透过这个镜子看到类的结构,所以称为反射!

Java反射机制可以完成

  1. 在运行时判断任意一个对象所属的类
  2. 在运行是构造任意一个类的对象
  3. 在运行时得到任意一个类所具有的成员变量和方法
  4. 在运行时调用任意一个对象的成员变量和方法
  5. 生成动态代理

反射相关的主要类

  1. java.lang.Class代表一个类,Class对象标识某个类加载后在堆中的对象
  2. java.lang.reflect.Method代表类的方法,Method对象标识某个类的方法
  3. java.lang.reflect.Field代表类的成员变量,Field对象表示某个类的成员变量
  4. java.lang.reflect.Constructor代表类的构造方法

示例

Cat.java

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16

public class Cat {
private String name = "招财猫";
public int age = 10; //public的
public Cat() {} //无参构造器
public Cat(String name) {
this.name = name;
}
public void hi() { //常用方法
//System.out.println("hi " + name);
}
public void cry() { //常用方法
System.out.println(name + " 喵喵叫..");
}
}

Reflection

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
//使用properties类,读写配置文件
Properties properties = new Properties();
properties.load(new FileInputStream("src\\re.propertoes"));
String classfullpath = properties.get("classfullpath").toString();
String methodName = properties.get("method").toString();
//使用反射机制
//加载类,返回class类型的对象
Class cls = Class.forName(classfullpath);
//通过cls得到加载的类com.Cat的对象实例
Object o = cls.newInstance();
System.outprint("o的运行类型="+o.getClass());
//在反射中将方法视为对象
Method method1 = cls.getMethod(methodName);
//通过method1调用方法
method1.invoke(o);//方法.invoke(对象);
Field nameField = cls.getField("age");
System.out.println(nameField.get(o));//成员变量.get(对象);
//构造器
Constructor con = cls.getConstructor();
//有参
Constructor con2 = cls.getConstructor(String.class);

反射优点和缺点

优点:可以动态的创建和使用对象也是框架底层核心,使用灵活,没有反射机制,框架技术就是去底层支撑。

缺点:使用反射基本是解释执行,对执行速度有影响

反射调用优化-关闭访问检查

  1. Method和Field、Constructor对象都有setAccessible()方法
  2. setAccessible作用是启动和禁用访问安全检查的开关
  3. 参数值为true表示反射的对象在使用时取消访问检查,提高反射的效率

Class类

  1. Class也是类,因此继承Object类
  2. Class类对象不是new出来的,而是系统创建的
  3. 对应某个类的Class类对象,在内存中只有一份,因为类只加载一次
  4. 每个类的实例都会记得自己是由哪个Class实例生成
方法名 说明
static Class forName(String name) 返回指定类名name的Class对象
Object newInstance() 调用缺省构造函数,返回gaiClass对象的一个实例
getName() 返回此Class对象所表示的实体名称
Class [] getInterfaces() 获取当前Class对象接口
ClassLoader getClassLoader() 返回该类的类加载器
Constructor[] getConstructors() 返回一个包含某些Constructor对象的数组
Method getMethod(String name,paramType) 返回一个Method对象,形参类型为paramType
Field [] getDeclaredFields() 返回Field对象的一个数组
Class getSuperclass() 返回表示此Class所表示的实体的超类Class