资讯专栏INFORMATION COLUMN

《深入理解JVM》读书笔记

idealcn / 1758人阅读

摘要:抽时间重新读了一遍深入理解一书。验证确保文件的字节流中包含的信息符合当前虚拟机的要求,并且不会危害虚拟机自身的安全。可见性可见性是指当一个线程修改了共享变量的值,其他线程能够立即得知这个修改。

抽时间重新读了一遍《深入理解JVM》一书。以下为摘录内容。

1 java内存区域

java虚拟机运行时数据区

1.1 程序计数器

是一块较小的内存空间,可以看做是当前线程所执行的字节码的行号指示器。每条线程都有一个独立的程序计数器,各条线程之间计数器互不影响。

1.2 java虚拟机栈

描述的是java方法执行的内存模型:每个方法在执行的同时都会创建一个栈帧用于存储局部变量表、操作数栈、动态链接、方法出口等信息。

局部变量表存放了编译器可知的各种基本数据类型、对象引用和returnAddress类型。

1.3 本地方法栈

虚拟机栈为虚拟机执行java方法服务,二本地方法栈为虚拟机使用到的Native方法服务。

1.4 java堆

被所有线程共享的一块内存区域,在虚拟机启动时创建。java堆是垃圾收集器管理的主要区域,因此很多时候也被叫做GC堆。

1.5 方法区

各个线程共享的内存区域,用于存储已被虚拟机加载的类信息、常量、静态变量、即时编译器编译后的代码等数据。

1.6 运行时常量池

方法区的一部分。需要注意的是string的intern方法在jdk1.6前后的不同。jdk1.6之后常量池放到了堆中。

1.7 直接内存

并不是虚拟机运行时数据区的一部分,也不是java虚拟机规范中国定义的内存区域。NIO引入的通道和缓冲区可以使用native函数库直接分配对外内存。

2 垃圾收集器与内存分配策略 2.1 判断对象是否存活的算法:

引用计数算法:很难解决对象之间相互循环引用的问题

可达性分析算法:通过一系列GC Roots的对象作为起始点,从这些节点开始向下搜索,搜索所走过的路线称为引用链,当一个对象到GC Roots没有任何引用链相连时,则证明此对象是不可用的。

2.2 垃圾收集算法

标记-清除算法:效率低,空间碎片化

复制算法:运行简单高效,代价高,降低了一半的使用率

标记-整理算法

分代收集:新生代用复制算法,老年代用标记整理算法

3 虚拟机类加载机制

加载、验证、准备、解析、初始化。

3.1 有且只有5种情况必须立即对类进行初始化

1)遇到new、getstatic、putstatic或invokestatic这4条指令字节码时,如果类没有进行过初始化,则需要先触发其初始化。

2)使用java.lang.reflect包的方法对类进行反射调用的时候,如果类没有进行过初始化,则需要先触发其初始化。

3)当初始化一个类的时候,如果发现其父类还没有进行过初始化,需要先触发其父类的初始化。

4)当虚拟机启动时,需要制定main,虚拟机会先初始化main类。

5)当使用jdk1.7的动态语言支持时,如果java.lang.invoke.MethodHandle实例最后的解析结果REF_getStatic、REF_putStatic、REF_invokeStatic的方法句柄,并且这个方法句柄所对应的类没有进行过初始化,则需要先触发其初始化。

3.2 类加载的过程 3.2.1 加载

1)通过一个类的全限定名来获取定义此类的二进制字节流

2)将这个字节流所代表的静态存储结构转换为方法区的运行时数据结构

3)在内存中生成一个代表这个类的java.lang.Class对象,作为方法区这个类的各种数据的访问入口。

3.2.2 验证

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

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

3.2.3 准备

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

3.2.4 解析

解析阶段是虚拟机将常量池内的符号引用替换为直接引用的过程。

类和接口的解析、字段解析、类方法解析、接口方法解析

3.2.5 初始化

类初始化阶段是类加载过程的最后一步。在准备阶段,变量已经赋过一次系统要求的初始值,而在初始化阶段,则根据程序员通过程序指定的主观计划去初始化类变量和其他资源,或者可以从另一个角度来表达:初始化阶段是执行类构造器()方法的过程。

4 java内存模型

硬件的效率与一致性

java内存模型(JMM)

线程、主内存、工作内存之间的交互关系
java内存模型规定了所有的变量都存储在主内存中,每条线程有自己的工作内存,线程的工作内存中保存了被该线程使用到的变量的主内存副本拷贝,线程对变量的所有操作(读取、赋值)都必须在工作内存中进行,而不能直接读写主内存中的变量。线程间变量值的传递均需要通过主内存来完成。

java内存模型时围绕着在并发过程中如何处理原子性、可见性和有序性这三个特征来建立的。

原子性:通过read、load、assign、user、store、write操作来保证。通过lock和unlock也可以满足。

可见性:可见性是指当一个线程修改了共享变量的值,其他线程能够立即得知这个修改。Java内存模型是通过在变量修改后将新值同步回主内存,在变量读取前从主内存刷新变量值这种依赖主内存作为传递媒介的方式来实现可见性的,无论普通变量还是volatile变量都是如此,普通变量与volatile变量的区别是,volatile的特殊规则保证了新值能够立即同步到主内存,以及每次使用前立即从主内存刷新。volatile保证了多线程操作时变量的可见性,二普通变量不能保证这一点。(synchronized和final关键字)

有序性:volatile和synchronized保证线程之间操作的有序性,volatile本身就包含了禁止指令重排序的语义。

先行发生原则 保证了我们大多数情况下不用关心太多。

文章版权归作者所有,未经允许请勿转载,若此文章存在违规行为,您可以联系管理员删除。

转载请注明本文地址:https://www.ucloud.cn/yun/73343.html

相关文章

  • 深入理解JVM虚拟机》读书笔记-开篇

    摘要:年开始工作,年毕业,两年来的工作接触知识面很广,用的东西比较多,包括基础的开发到开发到大数据,推荐系统,到服务器运维,到数据库维护,,,可愈发明白贪多嚼不烂的道理,唯有才能踏踏实实,趁着刚刚读完这本书,想复习,顺便写一些笔记,聊以巩固。 13年开始工作,14年毕业,两年来的工作接触知识面很广,用的东西比较多,包括基础的java开发到j2ee,web开发,到大数据,推荐系统,到服务器运维...

    legendaryedu 评论0 收藏0
  • 那些年我看过的书 —— 致敬我的大学生活 —— Say Good Bye !

    摘要:开头正式开启我入职的里程,现在已是工作了一个星期了,这个星期算是我入职的过渡期,算是知道了学校生活和工作的差距了,总之,尽快习惯这种生活吧。当时是看的廖雪峰的博客自己也用做爬虫写过几篇博客,不过有些是在前人的基础上写的。 showImg(https://segmentfault.com/img/remote/1460000010867984); 开头 2017.08.21 正式开启我...

    xiaoqibTn 评论0 收藏0
  • Java进阶之路

    摘要:探索专为而设计的将探讨进行了何种改进,以及这些改进背后的原因。关于最友好的文章进阶前言之前就写过一篇关于最友好的文章反响很不错,由于那篇文章的定位就是简单友好,因此尽可能的摒弃复杂的概念,只抓住关键的东西来讲,以保证大家都能看懂。 周月切换日历 一个可以进行周月切换的日历,左右滑动的切换月份,上下滑动可以进行周,月不同的视图切换,可以进行事件的标记,以及节假日的显示,功能丰富 Andr...

    sushi 评论0 收藏0
  • JVM从小白学成大佬】1.开篇

    摘要:的重要性毋庸置疑,可以毫不夸张的说虚拟机是整个平台的基石。方面的知识,也一直是等大厂面试考核的重点。本专栏将分为如下几个大模块进行分析开篇介绍运行时数据区。最主要的是让我知道能得到多少小伙伴的认可,毕竟大家的认可,就是不懈努力的动力 JVM的重要性毋庸置疑,可以毫不夸张的说Java虚拟机是整个Java平台的基石。 JVM方面的知识,也一直是BAT等大厂面试考核的重点。特别是JVM调优,...

    姘存按 评论0 收藏0
  • 对比JVM GC算法的读书笔记

    摘要:但是为了丰富引用的种类,以适应各种应用,中加入了中引用,但是除了强引用,其生命周期会有所不同,生存能力递减。加载该类的已被回收。 GC面临的问题有三个:哪些内存需要回收、什么时候回收和怎么回收 哪些内存需要回收,一般有两种方法 引用计数 对每个对象都有个被引用的次数,单被引用的次数为0的时候,就表示对象需要被回收 引用计数的缺点是没有办法解决循环引用导致的内存泄露问题 ...

    FrozenMap 评论0 收藏0

发表评论

0条评论

最新活动
阅读需要支付1元查看
<