JVM学习笔记(一)——类加载机制

一、初识JVM

​ 我们写好一份Java代码,要将其部署到线上的机器去运行,就要将其打包成.jar或者.war后缀的包,再进行部署。其中关键的一步是编译,也就是要把.java文件编译成.class字节码文件,有了字节码文件可以通过Java命令来启动一个JVM进程,由JVM来负责运行这些字节码文件。所以说,在某个机器上部署某个系统后,一旦启动这个系统,实际上就是启动类JVM

​ 我们写好一个个类是通过类加载器把字节码文件加载到JVM中的,JVM会首先从main()方法开始执行里面的代码,它需要哪个类就会使用类加载器来加载对应的类,反正对应的类就在.class文件中。

注意:如果一个项目中有多个main()方法,在启动一个.jar包的时候,就制定了是走哪个main()方法,所以入口是唯一的。

image-20200419212446836.png

程序运行机制

二、初识JVM类加载器机制

2.1 引入

问题:JVM什么时候会加载一个类?

最简单的例子是直接从main()开始执行,比如:

public class kafka{
    public static void main(String[] args){
        
    }
}
image-20200419213029027.png

如果碰到了实例化对象的操作,才把实例化的这个类的.class文件加载到内存(之前是没有加载进来的)

public class Kafka{
    public static void main(String[] args){
        ReplicaManager replicaManager = new ReplicaManager();
    }
}
image-20200419213505093.png

首先是包含main()方法的主类会在JVM启动之后首先被加载到内存中,然后开始执行main()方法中的代码,碰到需要使用的类,才去加载这个类对应的字节码文件,也就是说按需加载。

2.2 类加载过程

类加载的过程为:加载、验证、准备、解析、初始化、使用、卸载。

2.2.1 加载

加载是类加载的一个阶段,加载过程完成以下三件事:

  1. 通过类的完全限定名称获取定义该类的二进制字节流。
  2. 将该字节流表示的静态存储结构转换为方法区的运行时存储结构。
  3. 在内存中生成一个带表该类的Class对象,作为方法区中该类各种数据的访问入口。

2.2.2 验证

确保Class文件的字节流包含的信息符合当前虚拟机的要求,并且不会危害虚拟机自身的安全。

2.2.3 准备

分配内存空间,其次为类变量分配内存空间,并设定一个默认值。不执行赋值!

类变量是被static修饰的变量,准备阶段为类变量分配内存并设置初始值,使用的是方法区的内存。

实例变量不会在这阶段分配内存,它会在对象实例化时随着对象一起被分配在堆中。应该注意到,实例化不是类加载的一个过程,类加载发生在所有实例化操作之前,并且类加载只进行一次,实例化可以进行多次。

初始值一般为0值,例如下面的类变量value被初始化为0,而不是123。

public static int value = 123;

如果类变量是常量,那么它将初始化为表达式所定义的值而不是0。例如下面的常量value被初始化为123而不是0。

public static final int value = 123;

2.2.4 解析

将常量池的符号引用替换为直接引用的过程。

  1. 符号引用(Symbolic References): 符号引用以一组符号来描述所引用的目标,符号可以是任何形式的字面量,只要使用时能够无歧义的定位到目标即可。例如,在Class文件中它以CONSTANT_Class_infoCONSTANT_Fieldref_infoCONSTANT_Methordref_info等类型的常量出现。符号引用与虚拟机的内存布局无关,引用的目标并不一定加载到内存中。在Java中,一个Java类将会编译成一个Class文件。在编译时,Java类并不知道所引用的类的实际地址,因此只能使用符号引用类代替。比如org.simple.People类引用了org.simple.Language,在编译时People类并不知道Language类的实际内存地址,因此只能使用符号org.simple.Language(假设是这个,当然实际中是由类似于CONSTANT_Class_info的常量来表示的)来表示Language类的地址。各种虚拟机实现的内存布局可能有所不同,但是它们能接受的符号引用都是一致的,因为符号引用的字面量形式明确定义在Java虚拟机规范的Class文件格式中。

  2. 直接引用:直接引用可以是

  • 直接指向目标的指针(比如,指向“类型”【Class对象】、类变量、类方法的直接引用可能是指向方法区的指针)
  • 相对偏移量(比如,指向实例变量,实例方法的直接引用都是偏移量)
  • 一个能间接定位到目标的句柄

直接引用是和虚拟机布局相关的,同一个符号引用在不同的虚拟机实例上翻译出来的直接引用一般不会相同。如果有了直接引用,那引用的目标必定已经被加载到内存中了。

2.2.5 初始化

正式执行类初始化的代码,在这里才是执行赋值代码等操作,准备阶段仅仅为类和变量开辟空间。在这个阶段执行初始化操作有很多,比如对于静态代码块的初始化就是在这个阶段执行的(JVM设计者设计先执行静态代码块的机制就是希望开发者把类使用之前的准备工作在这准备好类级别的数据)。要记住,类的初始化就是初始化这个类,和里头的对象无关,只有new关键字才会构造一个对象。

什么时候会初始化一个类呢?

一般来说包含main()方法的类是必须立马初始化的,或者说执行到new对象了,就会把这个对象的类初始化,如果这个类初始化过了,就不用进行第二次初始化。初始化重要的一个规则是:初始化一个类的时候,如果该类的父类没有初始化,(如果父类也没有加载的话)必须先加载并初始化它的父类!

public class ReplicaManager extends AbstractDataManager{
    //ReplicaManager继承AbstractDataManager,在初始化ReplicaManager时必须先初始化它的父类
}

2.3 类加载器和双亲委派机制

2.3.1 Java中的类加载器

  • 启动类加载器:负责加载机器上安装的Java目录下的核心类,Java安装目录下有个lib文件夹存放了Java的核心库,JVM启动后,首先会依托启动类加载器去加载lib
  • 扩展类加载器:就是加载lib/ext目录,和启动类加载器差不多,但它是启动类加载器的儿子。
  • 应用程序类加载器:负责加载ClassPath环境变量指定路径中的类,就是把写好的代码加载进内存。
  • 自定义类加载器:自己写的类加载器,继承ClassLoader类,重写类加载方法。
image-20200420101220306.png

2.3.2 双亲委派机制

JVM的加载器是有亲子结构的,如图所示,提出了双亲委派机制

双亲委派机制:如果应用程序要加载一个类,首先会委派自己的父类加载器去加载,直至传到最顶层的加载器去加载,如果父类加载器在自己的职责范围内没有找到这个类,就会把加载权力下放给子类加载器。总的来说,就是先找父类去加载,不行再由儿子来加载。先从顶层加载器开始,发现自己加载不到,往下推给子类,这样能保证绝不会重复加载某个类

双亲委派的好处:避免了类的重复加载,如果两个不同层级的类加载器可以加载同一个类,就重复了。这使得Java类随着它的类加载器一起具有一种带有优先级的层次关系,从而使得基础类得到统一。

例如java.lang.Object存放在rt.jar中,如果编写另外一个java.lang.Object并放到ClassPath中,程序可以编译通过。由于双亲委派模型的存在,所以在rt.jar中的Object比在ClassPath中的Object优先级更高,这是因为rt.jar中的Object使用的是启动类加载器,而ClassPath中的Object使用的是应用程序类加载器。rt.jar中的Object优先级更高,那么程序中所有的Object都是这个Object

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