Java类加载机制

类加载器

类加载器的作用

类加载器(class loader)用来加载Java类到Java 虚拟机中。

一般来说,Java虚拟机使用Java类的方式如下:

  1. Java源程序(.java 文件)在经过Java编译器编译之后就被转换成Java字节代码(.class 文件)。
  2. 类加载器负责读取Java字节代码,并转换成java.lang.Class类的一个实例。

每个这样的实例用来表示一个 Java 类。通过此实例的 newInstance()方法就可以创建出该类的一个对象。实际的情况可能更加复杂,比如 Java 字节代码可能是通过工具动态生成的,也可能是通过网络下载的。

JVM默认的加载器
  • Bootstrap(引导类加载器):它用来加载Java的核心库,是用原生代码来实现的,并不继承自java.lang.ClassLoader。
  • ExtClassLoader(扩展类加载器):它用来加载Java的扩展库。Java虚拟机的实现会提供一个扩展库目录。该类加载器在此目录里面查找并加载Java类。
  • AppClassLoader(系统类加载器):它根据Java应用的类路径(CLASSPATH)来加载 Java 类。一般来说,Java应用的类都是由它来完成加载的。可以通过ClassLoader.getSystemClassLoader()来获取它。

类加载器也是一个Java类,本身也需要其他类加载器加载,必须要一个不是Java类的类加载器加载,这个类加载器就是Bootstrap(根结点)。
类加载器采用父子关系的树结构进行组织,在实例化每个类加载器时都需要指定其父级类加载器或者采用JVM默认的类加载器。
类加载树如图所示:

类加载器树.png

代码验证

public class Main {
    public static void main(String[] args) {
        ClassLoader classLoader=Main.class.getClassLoader();
        while (null!=classLoader){
            System.out.println(classLoader.getClass().getName());
            classLoader=classLoader.getParent();
        }
    }
}

output:

sun.misc.Launcher$AppClassLoader
sun.misc.Launcher$ExtClassLoader

注释:class Main位于buildpath目录下,所以类加载器为AppClassLoader,AppClassLoader的父级加载器为ExtClassLoader,ExtClassLoader的父级加载器为Bootstrap,但是Bootstrap不是一个Java类,所以输出为null

Class ClassLoader

类名:
public abstract class ClassLoaderextends Object
JDK的定义

A class loader is an object that is responsible for loading classes. The class ClassLoader is an abstract class. Given the binary name of a class, a class loader should attempt to locate or generate data that constitutes a definition for the class. A typical strategy is to transform the name into a file name and then read a "class file" of that name from a file system.
Every Class object contains a reference to the ClassLoader that defined it.

简单来说就是:

  • 基本上所有的类加载器都是 java.lang.ClassLoader类的一个实例
  • ClassLoader根据一个指定的类的名称,找到或者生成其对应的字节代码,然后从这些字节代码中定义出一个Java类,即java.lang.Class类的一个实例。
  • 除此之外,ClassLoader还负责加载Java应用所需的资源,如图像文件和配置文件等。

ClassLoader中与加载类相关的部分方法:

以上方法来源于JDK8

方法 英文描述
ClassLoader getParent() Returns the parent class loader for delegation.
Class<?> findClass(String name) Finds a class with the specified binary name, loading it if necessary.
Class<?> loadClass(String name) Loads the class with the specified binary name.
Class<?> defineClass(String name, byte[] b, int off, int len) Converts an array of bytes into an instance of class Class.
void resolveClass(Class<?> c) Links the specified class.

注意:binary name为字节码文件名

类加载机制

首先类加载,是把class文件从硬盘读取到内存中。
类加载方式:

  1. 程序在运行过程中当遇到通过new等方式生成对象时,隐式调用类装载器加载对应的类到jvm中
  2. 通过class.forname()等方法,显式加载需要的类
  3. 通过ClassLoader.loadClass()方法动态加载

类加载的步骤:

image.png

  1. 装载:查找和导入class文件;

  2. 连接:

       (1)检查:检查载入的class文件数据的正确性;
    
       (2)准备:为类的静态变量分配存储空间;
    
       (3)解析:将符号引用转换成直接引用(这一步是可选的)  
    
  3. 初始化:初始化静态变量,静态代码块。

类加载器的委托机制:

  • 当JVM需要加载一个类时,到底选择哪个类加载器进行加载了?

       1)首先当前线程的类加器会去加载线程中的第一个类。
       2)如果类A引用了类B,JVM会使用加载A的类加载器加载类B。
       3)当然也可以调用直接Class<?> loadClass(String name)来制定某个类加载器去加载类
    
  • 当类加载器加载类时,会委托给父级类加载器加载,当所有的祖宗类加载器没有加载到类,才会调用发起者类加载器,如果还是加载不了就会抛出ClassNotFoundException。

Java类的动态加载:
添加JVM option
-verbose:class

public class ClassDynamicLoading {
    public static void main(String[] args) {
        System.out.println("休息");
        try {
            Thread.sleep(1000);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        new ClassA().print("ClassA第一次使用");
        System.out.println("休息");
        try {
            Thread.sleep(1000);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        new ClassB().print("ClassB第一次使用");
        new ClassA().print("ClassA第二次使用");
    }
}
class ClassA{
   static {
        System.out.println("static code");
    }
   public void print(String str){
       System.out.println(str);
   }
}
class  ClassB{
    public void print(String str){
        System.out.println(str);
    }
}

output:

休息
[Loaded ClassA from file:/Users/wangkui/Desktop/Java%e7%b1%bb%e5%8a%a0%e8%bd%bd%e6%9c%ba%e5%88%b6/_classloader/out/production/_classloader/]
static code
ClassA第一次使用
休息
[Loaded ClassB from file:/Users/wangkui/Desktop/Java%e7%b1%bb%e5%8a%a0%e8%bd%bd%e6%9c%ba%e5%88%b6/_classloader/out/production/_classloader/]
ClassB第一次使用
ClassA第二次使用

分析以上代码输出可知,类只会使用到时才会被加载,加载后再次使用不会被加载。静态代码块只会在类加载完成后执行一次。

参考文档

深入探讨 Java 类加载器
JDK 8.0

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,591评论 6 501
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,448评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,823评论 0 353
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,204评论 1 292
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,228评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,190评论 1 299
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,078评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,923评论 0 274
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,334评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,550评论 2 333
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,727评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,428评论 5 343
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,022评论 3 326
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,672评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,826评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,734评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,619评论 2 354

推荐阅读更多精彩内容