记录JAVA反射学习
1.类的加载概述
当程序要使用某个类时,如果该类还未被加载到内存中,则系统会通过加载,连接,初始化三步来实现对这个类进行初始化。
a.加载 :就是指将class文件读入内存,并为之创建一个Class对象。任何类被使用时系统都会建立一个Class对象。
b. 连接:
验证 是否有正确的内部结构,并和其他类协调一致
准备 负责为类的静态成员分配内存,并设置默认初始化值
解析 将类的二进制数据中的符号引用替换为直接引用
c.初始化
d:加载时机
创建类的实例
访问类的静态变量,或者为静态变量赋值
调用类的静态方法
使用反射方式来强制创建某个类或接口对应的java.lang.Class对象
初始化某个类的子类
直接使用java.exe命令来运行某个主类
2.类加载器的概述
负责将.class文件加载到内存中,并为之生成对应的Class对象。虽然我们不需要关心类加载机制。
a.类加载器的分类
Bootstrap ClassLoader 根类加载器
Extension ClassLoader 扩展类加载器
Sysetm ClassLoader 系统类加载器
b.类加载器的作用
Bootstrap ClassLoader 根类加载器
也被称为引导类加载器,负责Java核心类的加载
比如System,String等。在JDK中JRE的lib目录下rt.jar文件中
Extension ClassLoader 扩展类加载器
负责JRE的扩展目录中jar包的加载。
在JDK中JRE的lib目录下ext目录
Sysetm ClassLoader 系统类加载器
负责在JVM启动时加载来自java命令的class文件,以及classpath环境变量所指定的jar包和类路径
3.反射概述
JAVA反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;
对于任意一个对象,都能够调用它的任意一个方法和属性;
这种动态获取的信息以及动态调用对象的方法的功能称为java语言的反射机制。
要想解剖一个类,必须先要获取到该类的字节码文件对象。
而解剖使用的就是Class类中的方法,所以先要获取到每一个字节码文件对应的Class类型的对象。
三种方式:
a:Object类的getClass()方法,判断两个对象是否是同一个字节码文件
b:静态属性class,锁对象
c:Class类中静态方法forName(),读取配置文件
public static void main(String[] args) throws ClassNotFoundException {
Class clzz1 = Class.forName("test.Person");
Class clzz2 = Person.class;
Person p = new Person();
Class clzz3 = p.getClass();
System.out.println(clzz1 == clzz2);
System.out.println(clzz2 == clzz3);
}
4.反射获取类相关
a.反射获取类的构造
public static void main(String[] args) throws ClassNotFoundException, InstantiationException, IllegalAccessException, Exception, SecurityException {
Class clazz = Class.forName("test.Person");
Person p = (Person) clazz.newInstance(); //若有无参构造函数
System.out.println(p);
Constructor c = clazz.getConstructor(String.class,int.class); //获取有参构造函数
Person p2 = (Person) c.newInstance("小红",23);
System.out.println(p2);
}
b.反射获取类的变量
public static void main(String[] args) throws Exception {
Class clazz = Class.forName("test.Person");
Person p = (Person) clazz.newInstance();
p.setName("小明");
p.setAge(22);
// Field field = clazz.getField("name"); //获取不为私有的变量
// field.set(p, "小红");
Field field2 = clazz.getDeclaredField("age"); //强行获取私有变量
field2.setAccessible(true); //打开访问权限
field2.set(p, 25);
System.out.println(p);
}
c.反射获取类的方法
public static void main(String[] args) throws Exception {
Class clazz = Class.forName("test.Person");
Person p = (Person) clazz.newInstance();
Method m = clazz.getMethod("eat"); //获取无参方法
m.invoke(p); //执行
Method m2 = clazz.getMethod("eat",int.class); //获取有参方法
m2.invoke(p,10);
}
5.动态代理
在Java中java.lang.reflect包下提供了一个Proxy类和一个InvocationHandler接口,通过使用这个类和接口就可以生成动态代理对象。JDK提供的代理只能针对接口做代理。我们有更强大的代理cglib,Proxy类中的方法创建动态代理类对象
public static Object newProxyInstance(ClassLoader loader,Class<?>[] interfaces,InvocationHandler h)
最终会调用InvocationHandler的方法
public class MyInvocation implements InvocationHandler {
private Object target;
public MyInvocation(Object o) {
this.target = o;
}
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
System.out.println("输出在方法之前");
method.invoke(target, args);
System.out.println("输出在方法之后");
return null;
}
}
public class InvocationTest {
public static void main(String[] args) {
Apple apple = new Apple();
apple.print();
InvocationHandler ih = new MyInvocation(apple);
FruitAll p2 = (FruitAll) Proxy.newProxyInstance(apple.getClass().getClassLoader(), apple.getClass().getInterfaces(), ih);
p2.print();
}
}