Java&Android 基础知识梳理(5) - 类加载&对象实例化

一、概述

虚拟机的类加载机制定义:把描述类的数据从Class文件(一串二进制的字节流)加载到内存,并对数据进行校验、转换解析和初始化,最终形成被虚拟机直接使用的Java类型。

Java语言里,类型的加载、连接和初始化过程都是在程序运行期间完成的,Java里天生可以动态扩展的语言特性就是依赖运行期动态加载和动态连接这个特点实现的。

用户可以通过Java预定义的和自定义类加载器,让一个本地的应用程序可以在运行时从网络或其他地方加载一个二进制流作为程序代码的一部分。

二、类加载的时机

2.1 类加载包含那些阶段

类从被加载到虚拟机内存中开始,到卸载出内存,所经过的生命周期有:

  • 1.加载
  • 2.验证
  • 3.准备
  • 4.解析
  • 5.初始化
  • 6.使用
  • 7.卸载

其中2-4统称为连接,上面的过程有几个需要注意的点:

  • 加载、验证、准备、初始化、卸载这五个阶段按顺序按部就班地开始,在一个阶段执行的过程中有可能调用、激活另外一个阶段。
  • 解析阶段有可能在初始化之后开始,这是为了支持Java语言的运行时绑定

2.2 类加载触发的时机

有且仅有下面五种情况必须立即对类进行初始化:

  • 第一种:遇到new/getstatic/putstatic/invokestatic4条字节码指令时,如果类没有进行过初始化,则需要先触发其初始化,场景:
    • 使用new关键字实例化对象
    • 读取或设置一个类的静态字段(被final修饰,已在编译期把结果放入常量池的字段除外)
    • 调用一个类的静态方法
        //1.new关键字.
        LoadInvokeClass loadInvokeClass = new LoadInvokeClass();
        //2.访问静态变量
        int content = LoadInvokeClass.sContent;
        //3.调用静态方法.
        LoadInvokeClass.staticMethod();
  • 第二种:使用java.lang.reflect包的方法对类进行反射调用的时候,如果类没有进行过初始化,则需要先触发其初始化。
        try {
            Class<?> mClass = Class.forName("com.example.lizejun.repojavalearn.load.LoadInvokeClass");
        } catch (Exception e) { e.printStackTrace(); }
  • 第三种:当初始化一个类的时候,如果需要初始化其父类,但是发现父类没有初始化、那么需要先触发其父类的初始化。
        //其中LoadInvokeClass是LoadInvokeClassChild的父类.
        LoadInvokeClassChild classChild = new LoadInvokeClassChild();
  • 第四种:当虚拟机启动时,用户需要指定一个要执行的主类(包含main()方法),虚拟机会先初始化这个主类。
  • 第五种:使用JDK 1.7的动态语言支持时,如果一个java.lang.invoke.MethodHandle实例最后的解析结果REF_getStatic/REF_putStatic/REF_invokeStatic的句柄方法,并且这个方法句柄所对应的类没有进行过初始化,则需要先触发其初始化。

2.3 被动引用

2.2中谈到的都是主动引用,除此之外,所有引用类的方法都称为被动引用,而被动引用不会触发类的初始化

  • 类初始化时,如果父类没有被初始化,那么会先初始化父类,这一过程将一直递归到Object为止,但是不会去初始化它所实现的接口,即当我们初始化ClassChild的时候,只会先初始化ClassParent,但不会初始化ClassInterface
public interface ClassInterface {}

public class ClassParent implements ClassInterface {
    static {
        System.out.println("load ClassParent");
    }
}

public class ClassChild extends ClassParent {
    static {
        System.out.println("load ClassChild");
    }
}
  • 接口初始化时,不要求父接口全部初始化,只有真正用到了父接口的时候(如引用接口中定义的常量),那么才会初始化。
  • 当访问某个类的静态域时,不会触发父类的初始化或者子类的初始化,即使静态域被子类或子接口或者它的实现类所引用,我们给ClassChild添加一个静态属性,访问这个静态属性不会初始化ClassParent
public class ClassChild extends ClassParent {

    public static int sNumber;

    static {
        System.out.println("load ClassChild");
    }
}
  • 如果一个静态变量是编译时常量,则对它的引用不会引起定义它的类的初始化,如下面访问sNumber,那么不会引起ClassChild的实例化。
public class ClassChild extends ClassParent {

    public static final int sNumber = 2;

    static {
        System.out.println("load ClassChild");
    }
}
  • 通过数组定义来引用类,不会触发此类的初始化。
ClassChild[] children = new ClassChild[10];

三、类加载的过程

3.1 加载

在"加载"阶段,虚拟机需要完成以下三件事情:

  • 通过一个类的全限定名来获取定义此类的二进制字节流
  • 将这个字节流所代表的静态存储结构转化为方法区的运行时数据结构
  • 在内存中生成一个代表这个类的java.lang.Class对象,作为方法区这个类的各种数据的访问入口

3.2 验证

"验证"阶段的目的是为了确保Class文件的字节流中包含的信息符合当前虚拟机的要求,并且不会危害自身的安全,大致会完成下面四个阶段的校验动作:

  • 文件格式验证
  • 元数据验证
  • 字节码验证
  • 符号引用验证

3.3 准备

"准备"阶段是正式为类变量(被static修饰,而不是实例变量)分配内存并设置类变量初始值的阶段,这些变量所使用的内存都将在方法区中进行分配。

  • 对于static并且非final的类变量,将被初始化为数据类型的零值。
  • 对于staticfinal的类变量,在这个阶段就会被初始化为ConstantValue属性所指定的值。

3.4 解析

“解析”阶段是虚拟机将常量池的符号引用替换为直接引用的过程,包括:

  • 类或接口的解析
  • 字段解析
  • 类方法解析
  • 接口方法解析

3.5 初始化

根据程序员通过程序指定的主观计划去初始化类变量和其它资源,也就是执行类构造器<clinit>()方法的过程:

  • <clinit>方法是由编译器自动收集类中的所有类变量的赋值动作和静态语句块中的语句合并而成,顺序是由语句在源文件中出现的顺序决定的。静态语句块只能访问到定义在它之前的变量,对于定义在它后面的变量只能赋值不能访问。

  • <clinit>()方法与类的构造函数不同,它不需要显示地调用父类构造器,虚拟机会保证在子类的<clinit>()方法执行前,父类的<clinit>()方法已经执行完毕,因此在虚拟机中第一个杯知行的<clinit>()方法的类肯定是java.lang.Object

  • 父类的静态语句块要优先于子类的变量赋值操作。

  • 如果一个类中没有静态语句块,也没有对类变量的赋值操作,那么编译器可以不为这个类生成<clinit>()方法。

  • 接口不能接口中仅有变量初始化的赋值操作,但执行接口的<clinit>()方法不需要先执行父接口的<clinit>()方法,只有当父接口中定义的变量使用时,父接口才会初始化,另外,接口的实现类在初始化时也一样不会执行接口的<clinit>()方法。

  • 虚拟机会保证一个类的<clinit>()方法在多线程环境中被正确地加锁、同步。

四、类加载器

4.1 概念

类加载器用来“通过一个类的全限定名来获取描述此类的二进制字节流”。

4.2 类与类加载器

类加载器用于实现类的加载动作,除此之外,任意一个类,都需要由它加载它的类加载器和这个类本身一同确立其在Java虚拟机中的唯一性。

每一个类加载器,都拥有一个独立的类名称空间,比较两个类是否相等,只有在两个类由同一个类加载器加载的前提下才有意义。

相等代表类的Class对象的equals方法,isAssignableFrom方法,isInstance方法。

4.3 双亲委派模型

绝大部分Java程序都会用到以下三种系统提供的类加载器:

  • 启动类加载器
  • 扩展类加载器
  • 应用类加载器

类加载器之间的层次关系,称为类加载器的双亲委派模型,这个模型要求除了顶层的启动类加载器外,其余的类都应当有自己的父类加载器,一般使用组合来复用父加载器的代码。

双亲委派模型的工作过程:如果一个类加载器收到了类加载的请求,它首先不会去尝试加载这个类,而是把这个请求委派给父类加载器去完成,只有当父类加载器反馈自己无法完成这个加载请求时,子加载器才会尝试自己加载。

五、对象实例化

在类加载过程完毕后,如果需要进行实例化对象就需要经过一下步骤,按优先加载父类,再到子类的顺序执行:

  • 加载父类构造器
  • 为父类实例对象分配存储空间并赋值
  • 执行父类的初始化块
  • 执行父类构造函数
  • 加载子类加载器
  • 为子类实例对象分配存储控件并赋值
  • 执行子类的初始化块
  • 执行子类构造函数

我们用一个简单的例子:
其中ClassOther是一个单独的类:

public class ClassOther {

    public int mNumber;

    public ClassOther() {
        System.out.println("ClassOther Constructor");
    }

    public void setNumber(int number) {
        this.mNumber = number;
    }

    public int getNumber() {
        return mNumber;
    }
}

ClassChild则继承于ClassChild

public class ClassParent {

    {
        System.out.println("ClassParent before mClassParentContent");
    }

    private ClassOther mClassParentContent = new ClassOther(10);

    {
        System.out.println("ClassParent after mClassParentContent=" + mClassParentContent.mNumber);
    }

    public ClassParent(int number) {
        mClassParentContent.setNumber(number);
        System.out.println("ClassParent Constructor, mClassParentContent=" + mClassParentContent.mNumber);
    }


}

public class ClassChild extends ClassParent {

    {
        System.out.println("ClassChild before a");
    }

    private int mClassChildContent = 1;

    {
        System.out.println("ClassChild after mClassChildContent=" + mClassChildContent);
    }

    public ClassChild() {
        super(2);
        System.out.println("ClassChild Constructor");
    }
}

当我们实例化一个ClassChild对象时,调用的顺序如下:

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

推荐阅读更多精彩内容