java中关于ClassLoader的知识

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进行解释和初始化。

 

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值