反射机制是在运行时,对于任意一个类,都能够知道这个类的所有属性和方法,对于任意一个对象,都能够调用它的任意一个方法,在java中,只要给定类的名字,就可以通过反射机制来获得类的所有信息,这种动态获取的信息以及动态调用对象的方法的功能称之为java语言的反射机制。
获取Class对象的几种方式:
- Class.forName(“全类名”) Class clazz3 = Class.forName("pers.hanchao.reflect.common.User");
- 类名.Class Class clazz1 = User.class;
- 对象名.getClass() User user = new User(); Class clazz2 = user.getClass();
反射的优缺点:
优点:
能够在运行时动态获取类的实例,提高灵活性
与动态编译结合
缺点:
使用反射性能较低,需要解析字节码,将内存中的对象进行解析
与正常的java代码相比较,效率低。
- getMethod和getDeclaredField方法会比invoke和set方法耗时;
- 随着测试数量级越大,性能差异的比例越趋于稳定;
- 不要过于频繁地使用反射,大量地使用反射会带来性能问题;
- 通过反射直接访问实例会比访问方法快很多,所以应该优先采用访问实例的方式。
相对不安全,破坏了封装性(因为通过反射可以获得私有方法和属性)
反射应用场景:
加载配置文件 加载JDBC驱动
通过反射调用Servlet
大部分框架都会使用反射 注入属性 调用方法 实例化等等
用反射一般需要实现的类:
- Class:表示正在运行的java应用程序中的接口
- Field:提供有关类和接口的属性信息,以及对它的的动态访问权限
- Constructor:提供关于类的单个构造方法的信息以及它的访问权限
- Method:提供类或接口中某个方法的信息
//通过new实例化对象
User user = new User();
//通过反射实例化对象
Class userClass = User.class;
User user1 = (User) userClass.newInstance();
//通过反射实例化对象
Constructor constructor = userClass.getDeclaredConstructor();
User user2 = (User) constructor.newInstance();
Constructor constructor1 = userClass.getDeclaredConstructor(String.class,String.class);
User user3 = (User) constructor1.newInstance("李四","000000");
try {
// 根据给定的类名初始化类
Class catClass = Class.forName("com.qunar.fresh.wangchaowang.exam1.test.Cat");
// 实例化这个类
Object obj = catClass.newInstance();
// 获得这个类的所有方法
Method[] methods = catClass.getMethods();
// 循环查找想要的方法
for(Method method : methods) {
if("setName".equals(method.getName())) {
// 调用这个方法,invoke第一个参数是类名,后面是方法需要的参数
method.invoke(obj, "Tom" ); } }
// 调用指定方法
catClass.getMethod("shout").invoke(obj, null);
} catch (ClassNotFoundException e) {
e.printStackTrace();
} catch (InstantiationException e) {
e.printStackTrace();
} catch (IllegalAccessException e) {
e.printStackTrace();
} catch (InvocationTargetException e) {
e.printStackTrace();
} catch (NoSuchMethodException e) {
e.printStackTrace();
}