ClassLoader in Java
简单写写Java反射和ClassLoader,之前玩过反射,觉得很有趣,这里就很简单的总结点,为学习Spring3.x做准备。
1.Java反射
在Jdbc中我们通常首先会根据一个字符串加载特定数据库驱动类的字节码,如下:
Class.forName("com.mysql.jdbc.Driver");
Class.forName("oracle.jdbc.OracleDriver");
Class.forName("com.microsoft.sqlserver.jdbc.SQLServerDriver");
这里其实是用到了类加载器ClassLoader:
// ClassLoader loader = ReflectTest.class.getClassLoader();
ClassLoader loader = Thread.currentThread().getContextClassLoader();
然后可以进行一些操作了:
//获取声明的构造方法(任何权限修饰符的),此处为默认的无参构造,如要生成有参的,只需传入相应参数的类型Class对象
Constructor cons = clazz.getDeclaredConstructor();
//创建对象
Car car = (Car) cons.newInstance();
//根据传入方法的参数的类型,获取setBrand方法
Method setBrand = clazz.getMethod("setBrand", String.class);
//回调方法
setBrand.invoke(car, "红旗CA72");
//同理
Method setMaxSpeed = clazz.getMethod("setMaxSpeed", int.class);
setMaxSpeed.invoke(car, 200);
问题来了,如果我们获取一个类的方法或属性是非public的,我们访问一个非public的方法或属性时如果不加设置会抛出异常: java.lang.IllegalAccessException
我们只需要加一行代码:setAccessible(true),通常不这样做,有特殊要求时可以这么解决;
// 获取private属性
Field colorFld = clazz.getDeclaredField("color");
// 设置可访问
colorFld.setAccessible(true);
colorFld.set(pcar, "红色");
// 获得private声明的方法
Method driveMtd = clazz.getDeclaredMethod("drive");
// 设置可访问
driveMtd.setAccessible(true);
driveMtd.invoke(pcar, (Object[]) null);
Java反射机制在诸多的框架中应用很广泛,我们在开发中常用的也不多,如需要自己编写一些框架,底层是逃离不了对Java反射的运用的,在JDK中reflect包中提供了这些类和接口,详细的可以在API中找到。
上面的类加载器ClassLoader:说说它的工作机制
ClassLoader是一个重要的Java运行时环境组件,负责在运行时查找和载入Class字节码文件。类加载工作是有ClassLoader及其子类负责的。
JVM在运行时会产生三个ClassLoader,准确的说是两个,因为根装载器不是ClassLoader的子类,它使用C++编写的,所以在Java中不能看到,看一个小程序:
public class ClassLoaderTest {
public static void main(String[] args) {
//获取当前线程上下文的类加载器
ClassLoader loader = Thread.currentThread().getContextClassLoader();
// ClassLoader loader = ClassLoaderTest.class.getClassLoader();
System.out.println("Current loader:"+loader);//当前类加载器
System.out.println("Parent loader:"+loader.getParent());//父加载器
System.out.println("Grandparent loader:"+loader.getParent().getParent());
}
}
输出结果是:
Current loader:sun.misc.Launcher$AppClassLoader@6b97fd
Parent loader:sun.misc.Launcher$ExtClassLoader@1c78e57
Grandparent loader:null
@ClassLoader:根装载器,负责装载JRE的核心类库,如JRE目标下的charsets.jar和rt.jar等,在Java中访问不到,无法获得它的句柄,返回null。
@ExtClassLoader:是ClassLoader的子类,负责装载JRE扩展目录ext中的jar类包。
@AppClassLoader:是ExtClassLoader的子类,负责装载Classpath路径下的类包。
全盘负责委托机制:
全盘负责:当一个ClassLoader装载一个类时,除非显式地指定使用另一个ClassLoader,该类锁依赖及引用的类也由这个ClassLoader载入。
委托机制:先委托父装载器寻找目标,只有在找不到的情况下在从自己的类路径中查找并装载目标类,这样就避免了一个安全隐患,
譬如我一不小心自己写了一个String(开始阶段这样写过),委托机制或委托父装载器去装载String类,这样就不会装载我写的这个String类了。
补充一下ClassLoader几个常用的接口方法:
Class loadClass(String name) name参数指定类装载器需要装载类的名字,必须使用全限定类名,如com.baobaotao. beans.Car。
该方法有一个重载方法loadClass(Stringname ,boolean resolve),resolve参数告诉类装载器是否需要解析该类。在初始化类之前,应考虑进行类解析的工作,
但并不是所有的类都需要解析,如果JVM只需要知道该类是否存在或找出该类的超类,那么就不需要进行解析。
Class findSystemClass(String name) 从本地文件系统载入Class文件,如果本地文件系统不存在该Class文件,将抛出ClassNotFoundException异常。该方法是JVM默认使用的装载机制。
Class findLoadedClass(String name) 调用该方法来查看ClassLoader是否已装入某个类。如果已装入,那么返回java.lang.Class对象,否则返回null。如果强行装载已存在的类,将会抛出链接错误。
ClassLoadergetParent() 获取类装载器的父装载器,除根装载器外,所有的类装载器都有且仅有一个父装载器,ExtClassLoader的父装载器是根装载器,因为根装载器非Java编写,所以无法获得,将返回null。
以下用一个图描述类装载器与对象的关系:
Object.getClass() Class ---实例对象获得类字节码
Class.getClassLoader() ClassLoader ----类字节码对象获得类装载器
ClassLoader.loaderClass(string className) Class ----类装载器转载特定的类