反射: java的特性之一
1. 定义:反射机制是在运行时,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意个对象,都能 够调用它的任意一个方法。在java中,只要给定类的名字,就可以通过反射机制来获得类的所有信息。
这种动态获取的信息以及动态调用对象的方法的功能称为 Java 语言的反射机制
描述: 是一种动态的程序加载和运行的机制
2. 原理: 反射的核心是JVM, jvm初次加载class文件时, 如果发现了未经编译的数据,
则将class文件返还编译器, 由编译器进行二次编译.
反射的API:
核心: Class, 类对象
3. 获取类对象的3种方法:
1) 通过class关键字获取, 例: Class clazz = User.class;
2) 通过getClass()获取, 例: User user = new User(); Class clazz = user.getClass();
3) 通过全类名获取, 例: Class clazz = Class.forName("com.three.bean.User");
注: 检查异常, ClassNotFoundException
通过Class对象获取实例: Object obj = clazz.newInstance();
其它: Field, Method, Constructor, ...
4 、实现 Java 反射的类:
1 ) Class :表示正在运行的 Java 应用程序中的类和接口
注意: 所有获取对象的信息都需要 Class 类来实现。
2 ) Field :提供有关类和接口的属性信息,以及对它的动态访问权限。
3 ) Constructor :提供关于类的单个构造方法的信息以及它的访问权限
4 ) Method :提供类或接口中某个方法的信息
5 、反射机制的优缺点:
优点:
1 )能够运行时动态获取类的实例,提高灵活性;
2 )与动态编译结合
缺点:
1 )使用反射性能较低,需要解析字节码,将内存中的对象进行解析。
解决方案:
1 、通过 setAccessible(true) 关闭 JDK 的安全检查来提升反射速度;
2 、多次创建一个类的实例时,有缓存会快很多
3 、 ReflflectASM 工具类,通过字节码生成的方式加快反射速度
2 )相对不安全,破坏了封装性(因为通过反射可以获得私有方法和属性)
编程语言的分类: (是否可以在程序的运行阶段对程序内容进行改变)
动态语言: 可以. 灵活但不稳定
静态语言: 不可以. 稳定但缺乏灵活性
示例:
package com.three.test;
import java.lang.reflect.Field;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import com.three.bean.User;
public class Test6 {
public static void main(String[] args) {
/* 获取类对象 */
//类名.class
Class clazz = User.class;
//对象.getClass()
User user = new User(1001, "韩梅梅", "6666");
clazz = user.getClass();
//Class.forName("全类名")
try {
clazz = Class.forName("com.three.bean.User");
} catch (ClassNotFoundException e) {
e.printStackTrace();
}
/* 通过Class获取实例 */
Object obj = null;
try {
obj = clazz.newInstance();
} catch (InstantiationException e) {
e.printStackTrace();
} catch (IllegalAccessException e) {
e.printStackTrace();
}
/* 通过Class对象操作属性 */
//操作单个属性
try {
//获取属性对象
// Field userName = clazz.getField("userName"); //可以获取父类属性, 但有访问权限限制
Field password = clazz.getDeclaredField("password"); //只可以获取本类属性, 但没有访问权限限制
//设置访问权限
password.setAccessible(true);
//读取属性值
// System.out.println(password.get(user));
//为属性赋值
password.set(obj, "9999");
// System.out.println(password.get(obj));
} catch (NoSuchFieldException e) {
e.printStackTrace();
} catch (SecurityException e) {
e.printStackTrace();
} catch (IllegalArgumentException e) {
e.printStackTrace();
} catch (IllegalAccessException e) {
e.printStackTrace();
}
//操作属性数组
Field[] fields1 = clazz.getFields();
Field[] fields2 = clazz.getDeclaredFields();
// System.out.println("1:" + fields1.length);
// System.out.println("2:" + fields2.length);
// for(Field f : fields2) {
// System.out.println("\t" + f.getName());
// }
/* 通过class对象操作方法 */
//操作单个方法
try {
//获取指定名称和参数列表的方法
Method m1 = clazz.getMethod("setUserName", String.class);
Method m2 = clazz.getMethod("getUserName");
//使用方法
// System.out.println(m2.invoke(user));
// System.out.println(m2.invoke(obj));
m1.invoke(obj, "李雷");
// System.out.println(m2.invoke(obj));
} catch (NoSuchMethodException e) {
e.printStackTrace();
} catch (SecurityException e) {
e.printStackTrace();
} catch (IllegalAccessException e) {
e.printStackTrace();
} catch (IllegalArgumentException e) {
e.printStackTrace();
} catch (InvocationTargetException e) {
e.printStackTrace();
}
//操作方法数组
Method[] methods1 = clazz.getMethods();
Method[] methods2 = clazz.getDeclaredMethods();
System.out.println("1:" + methods1.length);
for(Method m : methods1) {
System.out.println("\t" + m.getName());
}
System.out.println("------------------------------------------");
System.out.println("2:" + methods2.length);
for(Method m : methods2) {
System.out.println("\t" + m.getName());
}
}
}