从一道面试题谈起
![[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-tU4ZjT7q-1622120262211)(https://note.youdao.com/yws/res/5031/54817522C70F45C49939EFF6B1AED0D6)]](https://i-blog.csdnimg.cn/blog_migrate/0bb0c36d4743705846c7ae9eccd3ecf8.png)
![[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-T5F2TXv6-1622120262217)(https://note.youdao.com/yws/res/5033/836316377A9C4FAE9C0160D2916261AE)]](https://i-blog.csdnimg.cn/blog_migrate/eb4a86608bf36ab45f4317d6bd929332.png)
第一个截图代码运行结果是8
第二个截图代码运行结果是9
jvms 2.4 2.5
指令集分类
- 基于寄存器的指令集
- 基于栈的指令集
Hotspot中的Local Variable Table = JVM中的寄存器
Runtime Data Area
![[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-PSPNyb3n-1622120262220)(https://note.youdao.com/yws/res/5041/F6AE0C8BBC52406BB92811F175EEE6B5)]](https://i-blog.csdnimg.cn/blog_migrate/f0d5f5517533e3ad8a48c1b49ccdf140.png)
PC 程序计数器
存放指令位置
虚拟机的运行,类似于这样的循环:
while( not end ) {
取PC中的位置,找到对应位置的指令;
执行该指令;
PC ++;
}
JVM Stack
![[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-etYzjvkA-1622120262226)(https://note.youdao.com/yws/res/5045/4B8964D8661E4CBCA7376163B0F5CFFF)]](https://i-blog.csdnimg.cn/blog_migrate/5fcba3209f3a8387876412263105e9d8.png)
- Frame - 每个方法对应一个栈帧
- Local Variable Table(局部变量表,类似于寄存器)
- Operand Stack(操作数栈)
对于long的处理(store and load),多数虚拟机的实现都是原子的
jls 17.7,没必要加volatile - Dynamic Linking
https://blog.youkuaiyun.com/qq_41813060/article/details/88379473
jvms 2.6.3 - return address
a() -> b(),方法a调用了方法b, b方法的返回值放在什么地方,b方法调用结束后应该回到哪个位置继续执行
![[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-B5fdnkZD-1622120262228)(https://note.youdao.com/yws/res/5050/09DD278CF97E49A4B13051B00F8EF7F1)]](https://i-blog.csdnimg.cn/blog_migrate/e4ed41f723ebc6185122c704d6a89ea7.png)
![[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-jUwQVih9-1622120262230)(https://note.youdao.com/yws/res/5055/2D36CE31BD7744BEA511F029E1AFC8BC)]](https://i-blog.csdnimg.cn/blog_migrate/2274f672c80ca2fff0c7db6c6b6e91ce.png)
![[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-EgtGihNU-1622120262231)(https://note.youdao.com/yws/res/5057/08720D72D458462C9B1E07E75E605E77)]](https://i-blog.csdnimg.cn/blog_migrate/5d3a55b58e2dbfd2075712f6db00c1d9.png)
![[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-CapuXFkf-1622120262233)(https://note.youdao.com/yws/res/5059/794F9CFD5B5E451E98CF586D9AB0E8F0)]](https://i-blog.csdnimg.cn/blog_migrate/8fcd0f2c6c31e4a14728cac88d7913d7.png)
Heap
Method Area
- Perm Space (<1.8)
字符串常量位于PermSpace
FGC不会清理
大小启动的时候指定,不能变 - Meta Space (>=1.8)
字符串常量位于堆
会触发FGC清理
不设定的话,最大就是物理内存
Runtime Constant Pool
Native Method Stack
Direct Memory
JVM可以直接访问的内核空间的内存 (OS 管理的内存)
NIO , 提高效率,实现zero copy
思考:
如何证明1.7字符串常量位于Perm,而1.8位于Heap?
提示:结合GC, 一直创建字符串常量,观察堆,和Metaspace
常用指令
store
load
pop
mul
sub
invoke
- InvokeStatic
- InvokeVirtual
- InvokeInterface
- InovkeSpecial
可以直接定位,不需要多态的方法
private 方法 , 构造方法 - InvokeDynamic
JVM最难的指令
lambda表达式或者反射或者其他动态语言scala kotlin,或者CGLib ASM,动态产生的class,会用到的指令
本文深入探讨了JVM的内存结构,包括PC程序计数器、JVMStack、Heap、MethodArea等区域的功能。同时,讲解了基于寄存器和栈的指令集,以及Hotspot中的LocalVariableTable。文章还提到了不同版本JVM中字符串常量池的位置变化,并讨论了Invoke系列指令在多态方法调用中的作用。此外,还涉及了内存管理和GC在不同内存区域的影响。
1139

被折叠的 条评论
为什么被折叠?



