1.ClassLoader基本概念
与C或者C++编写的程序不同,Java程序并不是一个可执行文件,而是由许多独立的类文件组成,每一个文件对应一个java类。此外,这些类文件并非全部装入内存而是根据需要逐渐载入。ClassLoader是JVM实现的一部分,ClassLoader包括bootstrap classloader(启动类加载器),ClassLoader在JVM运行的时候加载java核心API,以满足Java程序最基本的需求,这些需求中包括用户定义的ClassLoader:一个是EXTClassLoader,主要用来加载java的扩展API,也就是lib/ext中的类;一个是AppClassLoader,主要用来加载用户机器上classpath设置目录下的class类,通常在没有指定ClassLoader的情况下,程序员自定义的类就有AppClassLoader加载。
2.ClassLoader加载流程
当运行一个程序时,JVM启动,运行bootstrap classloader(启动类加载器)加载java核心API(其中包括加载EXTClassLoader和AppClassLoader),然后调用EXTClassLoader加载扩展API,最后调用AppClassLoader加载classpath目录下定义的class类,这就是一个程序最基本的加载流程。
3.ClassLoader中一些重要的方法
1)loadClass()
ClassLoader.loadClass()是ClassLoader的入口点,该方法的定义如下:
class loadClass(String name , boolean resolve);
name是指JVM需要的类名称,resolve参数表示该类是否需要解析。注意:并不是加载所有的类都需要解析,如果JVM只 需要知道该类是否存在或者找出该类的超类,就不需要解析。
2)defineClass()
defineClass方法接受由原始字节组成的数组,并把它转换成Class对象。原始数组包含如文件系统或者网络装入的数据。 defineClass管理JVM的许多复杂的实现层面--它把字节码分析成运行时的数据结构、校验有效性等。因为defineClass方法 被标记成final的,所以也不能覆盖它。
3)findSystemClass()
findSystemClass方法从本地文件系统装入文件。它在本地文件系统中寻找类文件,如果存在,就是要defineClass将原始字节转换成Class对象,以将该文件转换成类。当运行java程序时,这是JVM正常装入类的默认机制。对于定制的ClassLoader,只有在尝试其他方法装入类之后,再使用findSystemClass。这是因为ClassLoader是负责执行装入类的相关步骤,不负责所有类的所有信息。即使ClassLoader从远程的Web站点装入了某些类,仍需要在本地机器上装入大量的基本java库。而这些类库不是我们所关心的,所以要JVM以默认方式从本地文件系统装入它们,这就是findSystemClass的用途。
4)findLoadedClass()
findLoadedClass充当一个缓存:当请求loadClass装入类时,它调用该方法来查看ClassLoader是否已装入这个类,这样可以避免重新装入已存在类所造成的麻烦。
5)findClass()
loadclass默认实现调用这个新方法。findClass的用途包含ClassLoader的所有特殊代码,而无需复制其他代码。这个方法通过类的名字得到一个Class对象。
6)getSystemClassloader()
如果覆盖findClass或loadClass,getSystemClassLoader能以实际的ClassLoader对象来访问系统ClassLoader(而不是固定的从findSystemClass中调用它)。为了将类请求委托给父类ClassLoader,这个新方法允许ClassLoader获取它的父类ClassLoader。当时用特殊方法,定制ClassLoader不能找到类时,可以使用这种方法。
7)forName()
Class类中有一个静态方法forName,这个方法和ClassLoader中的loadClass方法的目的一样,都是用来加载class的,但是两者在作用上却有所区别。
Class clazz =new Class.forName("something");
或者
ClassLoader cl =Thread.currentThread().getContextClassLoader();
Class clazz = cl.loadClass("something");
Class.forName()调用Class.forName(name , initialize , loader);也就是Class.forName ("something"); 等同于Class. forName ("something" , true , CALLCLASS.class.getClassLoader());
第二个参数"true"是用于设置加载类的时候是否连接该类,true表示连接,否则就不连接。关于连接就是在JVM加载类的时候,需要经过三个步骤:装载、连接、初始化。装载就是找到相应的class文件,读入JVM,初始化就是class文件初始化。这里详述一下连接,连接分为三步:
第一步是验证class是否符合规格;
第二步是准备,就是为类变量分配内存的同时设置默认初始值;
第三步就是解释,而这是可选的,根据上面loadClass方法的第二个参数来判定是否需要解释,这里的解释是指根据类中的符号引用查找相应的实体,再把符号引用替换成一个直接引用的过程。
在Java API文档中,loadClass方法的定义是protected,也就是说,该方法是被保护的,而用户使用的方法是一个参数,一个参数的loadClass方法实际上就是调用了两个参数,第二个参数默认是false。因此,在这里可以看出通过loadClass加载类实际上就是加载的时候并不对该类进行解释,因此就不会初始化该类。而Class类的forName方法则相反,使用forName加载的时候就会将Class进行解释和初始化。