jvm_book/README.md
2020-10-12 09:48:46 +08:00

17 KiB
Raw Blame History

《深入理解Java虚拟机第3版

广告
我正在撰写的《凤凰项目:现代软件架构探索》:https://github.com/fenixsoft/awesome-fenix

这是一部以“架构师应该掌握哪些架构知识”为叙事主线的、完全免费的开源文档如对您有用望不吝给个Star


  本工程为《深入理解Java虚拟机第3版》书中的样例代码以方便读者自行测试。部分代码需要在特定的虚拟机版本、参数下运行在代码前均已“VM Args”的注释进行标注。

  书中的勘误也会在本文中持续更新读者可通过issues提交新的勘误如对新版有任何建议也欢迎以issues或任何其他您方便的形式提出。

勘误列表:

  • Page 70到了JDK 7的HotSpot已经把原本放在永久代的字符串常量池、静态变量等【移出】
    更正到了JDK 7的HotSpot已经把原本放在永久代的字符串常量池、静态变量等【移至Java堆中在4.3.1节也会通过实验验证这一点】

  • Page 70:在虚拟机栈(栈帧中的本地变量表)中引用的对象,譬如【各个线程被调用的方法堆栈中使用到的参数、局部变量、临时变量等】
    有同学反应太过拗口,修改为:在虚拟机栈(栈帧中的本地变量表)中的引用的对象,譬如【当前正在运行的方法所使用到的参数、局部变量、临时变量等】

  • Page 385代码清单10-18 第13行需修改方法名称以符合输出结果
    protected void BADLY_NAMED_CODE() { 修改为 protected void Test() {

  • Page 443Java内存模型的基础设计并未改变即使是【这四操作种】对普通用户阅读使用起来仍然是并不方便。
    更正Java内存模型的基础设计并未改变即使是【这四种操作】对普通用户阅读使用起来仍然是并不方便。

  • Page 467在【第10章】里我们讲解“final关键字带来的可见性”时曾经提到过这一点
    更正在【第12章】里我们讲解“final关键字带来的可见性”时曾经提到过这一点

  • Page 472在【第10章】中我们知道了主流Java虚拟机实现中Java的线程是映射到操作系统的原生内核线程之上的
    更正在【第12章】中我们知道了主流Java虚拟机实现中Java的线程是映射到操作系统的原生内核线程之上的

以下勘误内容已在第5次重印版2020-9-11日修正


  • Page 51以下代码清单2-2为HotSpot虚拟机代表Mark Word中的代码【markOop.cpp】注释片段
    更正以下代码清单2-2为HotSpot虚拟机代表Mark Word中的代码【markOop.hpp】注释片段。另下面代码清单2-2也应更正为“代码清单2-2 【markOop.hpp】片段”

  • Page 72:要真正宣告一个对象死亡,【至少要】经历两次标记过程
    更正:要真正宣告一个对象死亡,【最多会】经历两次标记过程

  • Page 78图3-2、图3-3原稿中第2行2列、3行2列、3行4列都是“可回收”对象的颜色排版重画的时候颜色搞错了麻烦编辑下次印刷中修改过来。

  • Page 79发生垃圾【搜集】时将Eden和Survivor中仍然存活的对象一次性过拷贝到另外一块Survivor空间上
    更正发生垃圾【收集】时将Eden和Survivor中仍然存活的对象一次性过拷贝到另外一块Survivor空间上

  • Page 94-XX:GCTimeRatio参数的值则应当是一个大于0小于100的整数也就是垃圾收集时间占总时间的比率【相当于是吞吐量的倒数】。
    更正:-XX:GCTimeRatio参数的值则应当是一个大于0小于100的整数也就是垃圾收集时间占总时间的比率【这个参数设定为N的话表示用户代码执行时间与总执行时间之比为NN+1】

  • Page 95Parallel Old是Parallel Scavenge收集器的老年代版本支持多线程【并发收集】
    更正Parallel Old是Parallel Scavenge收集器的老年代版本支持多线程【并行收集】

  • Page 109:一旦用户程序访问到归属于旧对象的内存空间就会产生自陷【中段】
    更正:一旦用户程序访问到归属于旧对象的内存空间就会产生自陷【中断】

  • Page 128JDK 9之前虚拟机运行在Server模式下的默认值打开此开关后使用【Parallel Scavenge + Serial OldPS MarkSweep的】收集器组合进行内存回收
    更正JDK 9之前虚拟机运行在Server模式下的默认值打开此开关后使用【Parallel】收集器组合进行内存回收

  • Page 215Class文件是一组以【8个字节】为基础单位的二进制流各个数据项目严格按照顺序紧凑地排列在文件之中中间没有添加任何分隔符这使得整个Class文件中存储的内容几乎全部是程序运行的必要数据没有空隙存在。当遇到需要占用【8个字节】以上空间的数据项时则会按照高位在前 的方式分割成若干个【8个字节】进行存储。
    更正Class文件是一组以【字节】为基础单位的二进制流各个数据项目严格按照顺序紧凑地排列在文件之中中间没有添加任何分隔符这使得整个Class文件中存储的内容几乎全部是程序运行的必要数据没有空隙存在。当遇到需要占用【单个字节】以上空间的数据项时则会按照高位在前 的方式分割成若干个【字节】进行存储。

  • Page 2593 monitorenter // 以【栈定】元素即f作为锁开始同步
    更正3 monitorenter // 以【栈顶】元素即f作为锁开始同步

  • Page 265上述代码运行之后只会输出“SuperClass init!”
    更正上述代码运行之后【除value的值外】只会输出“SuperClass init!”

  • Page 280成为了Java技术体系中一块重要的基石可谓是【失之桑榆收之东隅】。
    更正成为了Java技术体系中一块重要的基石可谓是【失之东隅收之桑榆】。

  • Page 307'a'除了可以代表一个【字符串】
    更正:'a'除了可以代表一个【字符】

  • Page 392:脚注:是诸多跑分软件的测试【用力】通常都属于第二种,如果不去支持跑分会显得成绩很不好看。
    更正:是诸多跑分软件的测试【用例】通常都属于第二种,如果不去支持跑分会显得成绩很不好看。

  • Page 432在Outline视图中找到创建理想图的方法是【greateGraph()】
    更正在Outline视图中找到创建理想图的方法是【createGraph()】这个笔误在432页一共有3处

  • Page 472脚注2由于本书的主题是Java虚拟机【和】不是Java并发编程
    更正由于本书的主题是Java虚拟机【而】不是Java并发编程

  • Page 480:对一些代码要求同步,但是【对被检测】到不可能存在共享数据竞争的锁进行消除
    更正:对一些代码要求同步,但是【虚拟机又检测】到不可能存在共享数据竞争的锁进行消除

以下勘误内容已在第4次重印版2020-6-10日修正


  • Page 25:即不能动态加载其他【编译器】不可知的代码和类库
    更正:即不能动态加载其他【编译期】不可知的代码和类库

  • Page 30JDK 7则【还完全】处于研发状态的半成品。
    更正JDK 7则【完全是】处于研发状态的半成品。

  • Page 38由于JDK12已EOLRedHat的GitHub上目前只维护8、11两个LTS和13这个正在半年支持期内的JDK版本JDK12的Repo已经关闭但从Git的History中挖掘出来是很容易的为了便于读者获取 CMakeLists.txt我上传了一份到本仓库中。此文件版权归原作者所有。编辑在更新勘误时可跳过此条

  • Page 48Java是一门面向对象的编程语言Java程序运行过程中【无时无刻都】有对象被创建出来
    更正Java是一门面向对象的编程语言Java程序运行过程中【每时每刻都】有对象被创建出来

  • Page 58-60代码清单2-4、2-5的输出在代码中定义的方法名是“stackLeak()”在输出日志中看到的堆栈的方法名是“leak()”。这个是代码粘贴后整理编辑过方法名,不影响意思表达。(方法名在日志、代码中出现次数较多,编辑处理这条勘误时请单独联系我提供更正详情)

  • Page 80HotSpot虚拟机里面关注吞吐量的Parallel 【Scavenge】收集器是基于标记-整理算法的。
    更正HotSpot虚拟机里面关注吞吐量的Parallel 【Old】收集器是基于标记-整理算法的。

  • Page 97并发回收时垃圾收集线程【只占用不超过25%】的处理器运算资源,并且会随着处理器核心数量的增加而下降。
    更正并发回收时垃圾收集线程【占用不少于25%】的处理器运算资源,并且会随着处理器核心数量的增加而下降。

  • Page 105图3-14中第3行G1收集器的Compact阶段应该为【浅色】这个我交上去的原图是正确的但编辑重新画图时候涂层了深色。请编辑更新新版的时候注意一下。

  • Page 106在回收时通过这张表格就可以得出哪些Region之间产生了【跨代引用】。
    更正在回收时通过这张表格就可以得出哪些Region之间产生了【跨Region的引用】。

  • Page 107图3-15中两个X的位置打错了应该在5行3列、3行1列处打X

  • Page 106降低了处理跨代指针时的记忆集维护消耗也降低了伪共享问题【见3.4.4节】)的发生概率。
    更正降低了处理跨代指针时的记忆集维护消耗也降低了伪共享问题【见3.4.5节】)的发生概率。

  • Page 134如果在Survivor空间中【相同年龄】所有对象大小的总和大于Survivor空间的一半
    更正如果在Survivor空间中【低或等于某个年龄的】所有对象大小的总和大于Survivor空间的一半

  • Page 134满足【同年】对象达到Survivor空间的一半规则
    更正满足【低于或等于某年龄的】对象达到Survivor空间的一半规则

  • Page 162-163代码中类名“SynAddRunalbe”中英文“Runalbe”为拼写错误应为“SynAddRunnable”代码中一共有4处图片中有1处。

  • Page 239Exceptions属性的作用是列举出方法中可能抛出的受查异常Checked 【Excepitons】
    更正Exceptions属性的作用是列举出方法中可能抛出的受查异常Checked 【Exceptions】

  • Page 265创建动作由字节码指令【newarray】触发
    更正创建动作由字节码指令【anewarray】触发

  • Page 272【JDK 7及之前】HotSpot使用永久代来实现方法区时实现是完全符合这种逻辑概念的【而在JDK 8及之后】类变量则会随着Class对象一起存放在Java堆中
    更正【JDK 7之前】HotSpot使用永久代来实现方法区时实现是完全符合这种逻辑概念的【而在JDK 7及之后】类变量则会随着Class对象一起存放在Java堆中

  • Page 274被访问类C是public的不与访问类D处于同一个模块但是被访问类C的模块允许【被访问类D】的模块进行访问。
    更正被访问类C是public的不与访问类D处于同一个模块但是被访问类C的模块允许【访问类D】的模块进行访问。

  • Page 312运行结果的第3行This 【gay】 has $2
    更正This 【guy】 has $2

  • Page 314脚注三最后一行具体可【常见】第11章关于方法内联的内容。
    更正具体可【参见】第11章关于方法内联的内容。

  • Page 461图12-6New->RunningRunning->Terminated这两个是单向箭头交给编辑的原稿上也还是单向的应该是排版修图时搞错变成双向了请编辑同学修正过来。

  • Page 462:【以前处理一个请求可以允许花费很长时间在单体应用中】,具有这种线程切换成本也是无伤大雅的
    语句不通,更正为:【在以前的单体应用中,处理一个请求可以允许花费很长时间】,具有这种线程切换成本也是无伤大雅的

  • Page 462现实的需求在迫使Java去研究新的解决方案【同】大家又怀念起以前绿色线程的种种好处
    更正为现实的需求在迫使Java去研究新的解决方案【此时】大家又怀念起以前绿色线程的种种好处

以下勘误内容已在第3次重印版2020-2-20日修正


  • Page 27到了JDK 10HotSpot又重构了Java虚拟机的垃圾收集器接口 Java Virtual Machine 【Compiler】 Interface
    更正到了JDK 10HotSpot又重构了Java虚拟机的垃圾收集器接口 Java Virtual Machine 【Garbage】 Interface

  • Page 37譬如【Eclipst】 CDT或者NetBeans来进行的话
    更正譬如【Eclipse】 CDT或者NetBeans来进行的话

  • Page 110:对象的读取、写入、对象的比较、【为对象哈希值计算】、用对象加锁等等
    更正:对象的读取、写入、对象的比较、【为对象计算哈希值】、用对象加锁等等

  • Page 113图3-19中【Larage】为笔误应为【Large】

  • Page 23813: iload_1 后的注释应该是字节码第14行的

    13: iload_1 // 保存x到returnValue中此时x=2
    14: istore 4

    改为:

    13: iload_1
    14: istore 4 // 保存x到returnValue中此时x=2

  • Page 254:算术指令用于对【两个操作数栈上的值】进行某种特定运算
    对语序修改以避免歧义:算术指令用于对【操作数栈上的两个值】进行某种特定运算

  • Page 25913 astore_3 后注释【Taget】为笔误应为【Target】

  • Page 265/266在266页正文中出现两次注释一其中第一个注释是265页才对应该是排版问题请编辑再版时注意。

  • Page 278代码清单7-5第一行注释给变量【复制】可以正常编译通过
    更正: 给变量【赋值】可以正常编译通过

  • Page 278代码清单7-5第二行注释这句编译器会提示“非法【向前】引用”
    更正: 这句编译器会提示“非法【前向】引用”

  • Page 290用在IntelliJ 【IDE】、Eclipse这些IDE上做HotSwap……
    更正: 用在IntelliJ 【IDEA】、Eclipse这些IDE上做HotSwap……

  • Page 312代码实例中出现三处【gay】譬如Father gay = new Son(); 均应为【guy】这个不影响代码运行只是不太雅观。

  • Page 317产生这种差别产生的根本原因是Java语言在编译期间【却】已将println(String)方法完整的符号引用。
    更正产生这种差别产生的根本原因是Java语言在编译期间【就】已将println(String)方法完整的符号引用。

  • Page 322由于注解中John Rose博客文章中的代码托管网站Kenai.com已经关闭为了便于读者获取INDY工具我上传了一份到本仓库中(源码在src/indify目录)。此文件版权归原作者John Rose所有。编辑在更新勘误时可跳过此条

  • Page 348:通常解决【类】问题有以下几种途径
    更正:通常解决【此类】问题有以下几种途径

  • Page 372譬如将【代码清单10-2】稍微修改一下变成下面代码清单10-7这个样子
    更正譬如将【代码清单10-4】稍微修改一下变成下面代码清单10-7这个样子

  • Page 396【图11-2】和【图11-3】都仅仅是描述了客户端模式虚拟机的即时编译方式
    更正【图11-3】和【图11-4】都仅仅是描述了客户端模式虚拟机的即时编译方式

以下勘误内容已在第2次重印版2019-12-27日修正


  • 前言部分读者反馈信箱understandjvm@gmail.com
    更正由于这个信箱由于一直只收未发刚印刷后收到Google的通知此账号已自动作废。而且根据Google规定作废后无法注册同名邮箱。下次重印将修改为本工程地址:https://github.com/fenixsoft/jvm_book

  • Page 9支持HTTP 2客户【单】API等91个JEP
    更正支持HTTP 2客户【端】API等91个JEP

  • Page 64在【代码清单2-8】里笔者借助了CGLib……
    更正在【代码清单2-9】里笔者借助了CGLib……

  • Page 368: 【ArrayList<int>】与ArrayList<String>其实是同一个类型
    更正【ArrayList<Integer>】与ArrayList<String>其实是同一个类型