1
加载类的三种方法
使用new关键字让类加载器按需求载入所需的类
java.lang.Class的forName()方法加载
java.lang.ClassLoader的loadClass()方法加载
2
Class的方法
public static Class forName(String className)
public static Class forName(String className, boolean initialize,ClassLoader loader)
参数说明:
className - 所需类的完全限定名
initialize - 是否必须初始化类(静态代码块的初始化)
loader - 用于加载类的类加载器
调用只有一个参数的forName()方法等效于 Class.forName(className, true, loader)。
ClassLoader的方法
public Class<?> loadClass(String name)
protected Class<?> loadClass(String name, boolean resolve)
参数说明:
resolve是用于设置加载类的时候是否连接该类
调用只有一个参数的loadClass()方法等效于 loadClass(className,false)
而且只有它是public的
由此可见
默认下 forName会初始化 loadClass不会初始化
这也为什么JDBC里只能用forName不用loadClass的原因
3
在JVM加载类的时候,需要经过三个步骤,装载、连接、初始化。装载就是找到相应的class文件,读入JVM,初始化就不用说了,最主要就说说连接。
连接分三步,第一步是验证class是否符合规格,第二步是准备,就是为类变量分配内存同时设置默认初始值,第三步就是解释,而这步就是可选的,根据上面loadClass方法的第二个参数来判定是否需要解释
4
ClassLoader loader = obj.getClass().getClassLoader();
Class clazzA = loader.loadClass(AAA);
5
a, Bootstrap ClassLoader/启动类加载器
主要负责jdk_home/lib目录下的核心 api 或 -Xbootclasspath 选项指定的jar包装入工作.
b, Extension ClassLoader/扩展类加载器
主要负责jdk_home/lib/ext目录下的jar包或 -Djava.ext.dirs 指定目录下的jar包装入工作
c, System ClassLoader/系统类加载器
主要负责java -classpath/-Djava.class.path所指的目录下的类与jar包装入工作.
b, User Custom ClassLoader/用户自定义类加载器(java.lang.ClassLoader的子类)
在程序运行期间, 通过java.lang.ClassLoader的子类动态加载class文件, 体现java动态实时类装入特性.
- 启动顺序是abcd。其中至少a不是用java写的
- abcd是继承关系,a是b的父类
- 在加载类的时候,采用 双亲委派 。
也就是说,如果是系统类加载器加载,那么先委派给扩展类加载器,扩展类加载器再委派给启动类加载器
如果启动类加载器找不到,在分发给扩展类加载器,扩展类加载器还找不到,再回到系统类加载器
参考
http://www.iteye.com/topic/83978