【JVM】JVM之类加载器

时间:2022-02-16 09:01:07

一、前言

  首先,小小测试,看是否已经掌握了JVM类加载的过程

  1.1、测试一

class Singleton {
private static Singleton sin = new Singleton();
public static int counter1;
public static int counter2 = 0; private Singleton() {
counter1++;
counter2++;
} public static Singleton getInstance() {
return sin;
}
} public class Test {
public static void main(String[] args) {
Singleton sin = Singleton.getInstance();
System.out.println(sin.counter1);
System.out.println(sin.counter2);
}
}

  输出结果为:

  1 3 ?

  1 0 ?  

  0 1?

  1.2、测试二:

class Singleton {
public static int counter1;
public static int counter2 = 2;
private static Singleton sin = new Singleton(); private Singleton() {
counter1++;
counter2++;
} public static Singleton getInstance() {
return sin;
}
} public class Test {
public static void main(String[] args) {
Singleton sin = Singleton.getInstance();
System.out.println(sin.counter1);
System.out.println(sin.counter2);
}
}

  

输出结果为:

  1 3 ?

  1 2 ?  

  0 1?

正确的输出结果如下:

测试一的结果为:1 0

测试二的结果为:1 3

如果对结果有疑惑或者不知道原因的园友需要了解类加载器的具体细节,相信看了本篇文章,一定会解开您的疑惑。废话不多说,直奔主题。

二、背景知识

  2.1、Java虚拟机通过装(加)载、连接、初始化一个Java类型,使该类型可以被正在运行的Java程序所使用。  

  ①装(加)载类的加载指的是将类的.class文件中的二进制数据读入到内存中,将其放在运行时数据区的方法区中,然后在堆区创建一个java.lang.Class对象,用来封装类在方法区内的数据结构,之后可以用Class对象进行相关的反射操作。

  ②连接分为三个子步骤    

    验证:确保被加载的类的正确性

    准备:为类的静态变量分配内存,并将其初始化为默认值

    解析: 把类中的符号引用转换为直接引用

  ③初始化为为类的静态变量赋予正确的初始值

  如下为流程图:【JVM】JVM之类加载器

  2.2、关于初始化的细节  

  所有的Java虚拟机实现必须在每个类或接口被Java程序“首次主动使用”时才初始化他们,下面六种情况符合首次主动使用要求。    

    ① 创建类的实例    

    ②访问某个类或接口的非常量静态域,或者对该非常量静态域赋值

    ③ 调用类的静态方法

    ④反射(如Class.forName(“com.test.Test”)(其中Test为一个类),而Test.class就不是首次使用)

    ⑤初始化一个类的子类

    ⑥Java虚拟机启动时被标明为启动类的类(Test)(Test为包含了程序入口main方法的类)

  2.3、如论如何,如果一个类型在其首次主动使用之前还没有被装(加)载和连接的话,那它必须在此时进行加载和连接,这样它才能够被初始化。

三、问题分析

  3.1、读到这里应该可以分析出为什么之前的程序会输出那样的结果,下面我们来一起分析一下整个过程。

  ①对于测试一的结果分析

  首先在main中调用了Singleton的getInstance类静态方法,符合第③条,需要初始化类,即初始化Singleton,首先需要装(加)载和连接,从硬盘中加载进内存,然后进入连接的验证,没有问题,OK,进入准备阶段,此时,将类变量sin、counter1、counter2分配内存,并初始化默认值null、0、0。紧接着,将符号引用转化为直接引用(暂  时不不要太了解,程序中就是将Singleton符号转化为在内存里直接对地址的引用,这样就可以通过地址直接访问Singleton类型了)。接下来是初始化过程,首先调用sin的构造方法,然后将counter1、counter2分别+1,即counter1 = 1,counter2 = 1,完成了sin静态变量的初始化,然后初始化静态变量counter1,但是由于没有对counter1赋初值,所以counter1还是为1,然而,程序中对counter2进行了赋初值操作,即将counter2赋值为0。这样便完成了类型的初始化,得到的counter1和counter2的结果为1和0,分析完毕。

  结果分析流程图如下:

    【JVM】JVM之类加载器

    

②同理也可以对测试二进行结果分析

四、总结

  整个类的初始化三个阶段细节过程远比这个复杂得多,但是我们仍可以通过类的整个宏观流程来分析出正确的结果,对过程的分析也有助于我们写出正确的程序。真正做到知其然知其所以然。也感谢各位园友的观看,谢谢。

【JVM】JVM之类加载器的更多相关文章

  1. Java虚拟机JVM学习05 类加载器的父委托机制

    Java虚拟机JVM学习05 类加载器的父委托机制 类加载器 类加载器用来把类加载到Java虚拟机中. 类加载器的类型 有两种类型的类加载器: 1.JVM自带的加载器: 根类加载器(Bootstrap ...

  2. JVM的艺术—类加载器篇(二)

    分享是价值的传递,喜欢就点个赞 引言 今天我们继续来深入的剖析类加载器的内容.上节课我们讲了类加载器的基本内容,没看过的小伙伴请加关注.今天我们继续. 什么是定义类加载器和初始化类加载器? 定义类加载 ...

  3. JVM的艺术—类加载器篇(三)

    JVM的艺术-类加载器篇(三) 引言 今天我们继续来深入的剖析类加载器的内容.上篇文章我们讲解了类加载器的双亲委托模型.全盘委托机制.以及类加载器双亲委托模型的优点.缺点等内容,没看过的小伙伴请加关注 ...

  4. Java虚拟机笔记 – JVM 自定义的类加载器的实现和使用2

    1.用户自定义的类加载器: 要创建用户自己的类加载器,只需要扩展java.lang.ClassLoader类,然后覆盖它的findClass(String name)方法即可,该方法根据参数指定类的名 ...

  5. JVM学习--(六)类加载器原理

    我们知道我们编写的java代码,会经过编译器编译成字节码文件(class文件),再把字节码文件装载到JVM中,映射到各个内存区域中,我们的程序就可以在内存中运行了.那么字节码文件是怎样装载到JVM中的 ...

  6. JVM启动过程 类加载器

    下图来自:http://blog.csdn.net/jiangwei0910410003/article/details/17733153 package com.test.jvm.common; i ...

  7. JVM学习记录-类加载器

    前言 JVM设计团队把类加载阶段中的“通过一个类的全限定名来获取描述此类的二进制字节流”这个动作放到Java虚拟机外面去实现,以便让应用程序自己决定如何去获取所需要的类.实现这个动作的代码模块称为“类 ...

  8. 【深入理解JVM】:类加载器与双亲委派模型

    类加载器 加载类的开放性 类加载器(ClassLoader)是Java语言的一项创新,也是Java流行的一个重要原因.在类加载的第一阶段“加载”过程中,需要通过一个类的全限定名来获取定义此类的二进制字 ...

  9. JVM体系结构之二:类加载器之2:JVM 自定义的类加载器的实现和使用

    一.回顾一下jdk自带的类加载器: 1.java虚拟机自带的加载器     根类加载器(Bootstrap,c++实现)     扩展类加载器(Extension,java实现)     应用类加载器 ...

  10. JVM概述和类加载器

    JVM概述 1.Java虚拟机所管理的内存包括以下几个运行时数据区域:   ①.程序计数器     程序计数器是一块较小的内存空间,可以看做是当前线程所执行的字节码的行号指示器,字节码解释器工作时就是 ...

随机推荐

  1. 解决 Error: getaddrinfo EADDRINFO 错误

    安装npm失败,提示Error: getaddrinfo EADDRINFO,原因在于虚拟机未连接互联网,悲剧.

  2. MySQL同步常见问题解答(自己的小心得)

    前几天刚刚注册了博客园,我想写一些技巧性的教程,今天给大家分享一个MySQL同步常见问题解答. Q:如果主服务器正在运行并且不想停止主服务器,怎样配置一个从服务器? A:有多种方法.如果你在某时间点做 ...

  3. 三层交换单臂路由vlan间通信综合实验之降龙要点[转]

    单臂路由三层交换机提供vlan间的通信之菜鸟之降龙详解要点: 图示 PC:左到右依次设置IP172.16.10.1,    20.1,  30.1,   40,1  ,50,1  /24 网关10.2 ...

  4. SQL技术内幕-13 SQL优化方法论之分析实例级别的等待

    优化方法论的第一步是在实例级别上找出什么类型的等待占用了大部分的等待时间,这可以通过查询动态管理图(DMV,dynamic management view)sys.dm_os_wait_stats 运 ...

  5. debian linux 中如何查看软件包是否已经安装和如何安装、卸载软件

    练习 1 方案:确定软件包是否安装 如果您不确定某个软件包是否已经安装,可以使用 dpkg 的 -l (L的小写) 选项: $ dpkg -l zsh No packages found matchi ...

  6. Unity 关于属性的get/set

    学习Unity的可能多数是C#转过来的, 一进来的时候你会发现Unity编写代码,在一些视频或文章中.基本都没有用过get/set使用, 多数是public string name;这样写的公开字段, ...

  7. Python with ASP

    Python with ASP Python with ASP

  8. SSH整合中为获取表单对象Action类实现的接口及拦截器配置

    保存员工或者用户信息时,以员工为例,是通过表单收集信息的,需要把这些信息赋给一个对象,然后保存到数据库中.对应的Action类须实现Preparable接口及ModelDriven接口,且在Actio ...

  9. 第一章 Python基本语法元素分析(二)

    1.3   实例1:温度转换 根据华氏和摄氏温度定义,利用转换公式如下: C=(F-32)/1.8 F=C*1.8+32 代码如下: 运行结果: 1.4   Python程序语法元素分析 注释:不被程 ...

  10. th:标签

    https://blog.csdn.net/xxb5502296/article/details/78319898(挺全的) https://blog.csdn.net/qq_43279637/art ...