概述
JAVA反射机制是在运行状态中,获取任意一个类的结构 , 创建对象 , 得到方法,执行方法 , 属性;这种在运行状态动态获取信息以及动态调用对象方法的功能被称为java语言的反射机制。
类加载器
Java类加载器(Java Classloader)是Java运行时环境(Java Runtime Environment)的一部分,负责动态加载Java类到Java虚拟机的内存空间中。
java默认有三种类加载器,BootstrapClassLoader、ExtensionClassLoader、AppClassLoader。
BootstrapClassLoader(引导启动类加载器):
嵌在JVM内核中的加载器,该加载器是用C++语言写的,主要负载加载JAVA_HOME/lib下的类库,引导启动类加载器无法被应用程序直接使用。
ExtensionClassLoader(扩展类加载器):
ExtensionClassLoader是用JAVA编写,且它的父类加载器是Bootstrap。是由sun.misc.Launcher$ExtClassLoader实现的,主要加载JAVA_HOME/lib/ext目录中的类库。它的父加载器是BootstrapClassLoader
App ClassLoader(应用类加载器): App
ClassLoader是应用程序类加载器,负责加载应用程序classpath目录下的所有jar和class文件。它的父加载器为ExtensionClassLoader
双亲委派模型:如果一个类加载器收到了一个类加载请求,它不会自己去尝试加载这个类,而是把这个请求转交给父类加载器去完成。每一个层次的类加载器都是如此。因此所有的类加载请求都应该传递到最顶层的启动类加载器中,只有到父类加载器反馈自己无法完成这个加载请求(在它的搜索范围没有找到这个类)时,子类加载器才会尝试自己去加载。委派的好处就是避免有些类被重复加载。
得到class的三种方式
public class Demo2 {
public static void main(String[] args) throws ClassNotFoundException {
//第一种方式 通过包名.类名.class 加载类
Class<Person> c1 = com.java.fanshe.Person.class;
System.out.println(c1);
//第二种方式 通过类的对象获取类得信息
Person p =new Person();
Class<Person> c2 = (Class<Person>) p.getClass();
System.out.println(c2);
System.out.println(c1 == c2);
//第三种方式
Class<Person> c3 = (Class<Person>) Class.forName("com.java.fanshe.Person");
System.out.println(c3);
System.out.println(c1 == c2 && c1 == c3);
}
}
结果如下:
class com.java.fanshe.Person
class com.java.fanshe.Person
true
class com.java.fanshe.Person
true
上述的三种方式, 在调用时, 如果类在内存中不存在, 则会加载到内存 ! 如果类已经在内存中存在, 不会重复加载, 而是重复利用
获取构造对象Constructor
1、通过指定的参数类型, 获取指定的单个构造方法
getConstructor(参数类型的class对象数组)
例如:
构造方法如下: Person(String name,int age)
得到这个构造方法的代码如下:
Constructor c = p.getClass().getConstructor(String.class,int.class);
2、获取构造方法数组
getConstructors();
3、 获取所有权限的单个构造方法
getDeclaredConstructor(参数类型的class对象数组)
4、 获取所有权限的构造方法数组
getDeclaredConstructors();
Constructor 创建对象
常用方法:
newInstance(Object... para)
调用这个构造方法, 把对应的对象创建出来
参数: 是一个Object类型可变参数, 传递的参数顺序 必须匹配构造方法中形式参数列表的顺序!
setAccessible(boolean flag)
如果flag为true 则表示忽略访问权限检查 !(可以访问任何权限的方法)
获取方法Method
- getMethod(String methodName , class… clss)
根据参数列表的类型和方法名, 得到一个方法(public修饰的)- getMethods();
得到一个类的所有方法 (public修饰的)- getDeclaredMethod(String methodName , class… clss)
根据参数列表的类型和方法名, 得到一个方法(除继承以外所有的:包含私有, 共有, 保护, 默认)- getDeclaredMethods();
得到一个类的所有方法 (除继承以外所有的:包含私有, 共有, 保护, 默认)
执行方法
invoke(Object o,Object… para) :
调用方法 ,
参数1. 要调用方法的对象
参数2. 要传递的参数列表
getName()
获取方法的方法名称
setAccessible(boolean flag)
如果flag为true 则表示忽略访问权限检查 !(可以访问任何权限的方法)
获取属性Field
- getDeclaredField(String filedName)
根据属性的名称, 获取一个属性对象 (所有属性)- getDeclaredFields()
获取所有属性- getField(String filedName)
根据属性的名称, 获取一个属性对象 (public属性)- getFields()
获取所有属性 (public)
import java.lang.reflect.Constructor;
import java.lang.reflect.Field;
public class Demo5 {
public static void main(String[] args) throws Exception {
Class c = Class.forName("com.java.fanshe.Person");
Constructor ct = c.getConstructor();
Object o = ct.newInstance();
Field phoneNumber = c.getField("phoneNumber");
phoneNumber.set(o,"13843838438");
//访问带权限的
Field name = c.getDeclaredField("name");
name.setAccessible(true);
name.set(o,"张三");
System.out.println(o);
}
}