-
每一个方法从调用直至执行完成的过程,就对应着一个栈帧在虚拟机栈中入栈到出栈的过程
-
栈帧是用来存储数据和部分过程结果的数据结构,同时也被用来处理动态链接(Dynamic Linking)、 方法返回值和异常分派(Dispatch Exception)。栈帧随着方法调用而创建,随着方法结束而销毁(无论方法是正常完成还是异常完成)
-
如果线程请求的栈深度大于虚拟机允许深度,则抛出StackOverflowError;扩展时无法申请到足够内存,则抛出OutOfMemeryError
-
2.3 本地方法栈(Native Method Stack)
-
本地方法栈和虚拟机栈作用类似,区别是虚拟机栈为执行Java方法服务,而本地方法栈则为Native方法服务。(HopShot的实现 直接把本地方法栈和虚拟机栈合二为一)
-
上述3类区域,生命周期与Thread相同,即:线程创建时,相应的内存区创建,线程销毁时,释放相应内存
-
2.4 堆(Heap)
-
线程共享的一块内存区域,几乎所有的对象实例在这里分配内存,也是垃圾收集器进行垃圾收集的最重要的内存区域。因此很多时候也叫GC堆
-
线程私有的分配缓存区(Thread Local Alloaction Buffer)也是在堆划分出来的
-
JDK8的版本,因使用元空间代替永久代,字符串常量池和类的静态变量也放入java堆中
-
2.5 元空间(MetaSpace)
-
主要存储类的元数据,比如类的各种描述信息,类名、方法、字段、访问限制等,既编译器编译后的代码等数据
-
运行时常量池:Class文件中除了有类的版本、字段、方法等描述等信息外;还有一项信息是常量池,用于存放编译期生成的各种字面量和符号引用,这部分将在类加载后存放到元空间的运行时常量池中
-
使用元空间代替永久代原因
-
永久代的大小是在启动时固定好的,很难进行调优;太大则容易导致永久代溢出;太小在运行时,容易抛出OutOfMemeryError
-
字符串存在永久代中,使用时易出问题,由于永久代内存经常不够用,爆出异常OutOfMemoryError: PermGen
-
CodeCache
-
JVM生成的native code存放的内存空间称之为Code Cache;JIT编译、JNI等都会编译代码到native code,其中JIT生成的native code占用了Code Cache的绝大部分空间
-
直接内存
-
它并不是虚拟机运行时数据区的一般分,也不在规范定义。JDK1.4,引入了Channel(通道)与Buffer(缓存区)的I/O方式,它可以使用Native函数分配堆外内存,可通过DirectByteBuffer操作。
3 JVM运行时内存布局和JMM内存模型区别
- JVM内存区域是指JVM运行时将内存数据分区域存储,强调对内存空间的划分
- JAVA内存模型是Java语言在多线程并发情况下对于共享变量内存操作的规范:解决变量在多线程的可见性、原子性的问题
4 JMM内存模型交互操作
-
内存交互操作有八种,虚拟机的实现保证每一个操作都是原子性的
-
lock(锁定):作用于主内存的变量,标识变量为线程独占状态
-
unlock(解锁):作用于主内存的变量,释放一个处于锁定状态的变量,释放后的变量才可以被其他线程锁定
-
read(读取):作用于主内存变量,从主内存中读取出后面load操作要用到的变量
-
load(载入):作用于主内存中的变量,把刚才read的值放入工作内存的副本中
-
use(使用):作用于工作内存中的变量,当线程执行某个字节码指令需要用到相应的变量时,把工作内存中的变量副本传给执行引擎
-
assign(赋值):作用于工作内存中的变量,把一个从执行引擎中接受到的值放入工作内存的变量副本中
-
store(存储):作用于工作内存中的变量,把工作内存中的变量送到主内存,给后续的write使用
-
write(写入):作用于主内存中的变量,把store的工作内存中的变量值,写入主内存中
-
read和load 好像是相同的操作?各位有何高见,请指教下
-
JMM对这八种指令的使用,制定了如下规则
-
read和load、store和write必须顺序执行,而且两个指令绑定出现;就是说出现read就要有load
-
不允许一个线程丢弃最近的assign操作,工作内存中的变量改变后,必须write同步到主内存
-
不允许一个线程把没有发生assign操作的变量同步到主内存
-
新的变量必须诞生于主内存,不允许工作内存使用一个没有初始化的变量;use、store操作变量之前,必须经过load和assign操作
-
变量同一时刻只允许一个线程对其lock,该线程可以对该变量加锁多次,释放锁需要执行相同次数的unlock,lock和unlock要成对出现
-
一个变量没有lock,不能unlock;并且一个线程不能unlock被其他线程锁住的变量
-
执行unlock前,必须把工作内存中的变量同步到主内存中
-
执行lock操作,需要清空工作内存(所有),并且需要使用该变量之前,要重新执行load和assign操作
#####大家看完有什么不懂的可以在下方留言讨论也可以关注.
#####谢谢你的观看。
#####觉得文章对你有帮助的话记得关注我点个赞支持一下!
链接:https://juejin.im/post/6861410796233195528
自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。
深知大多数Java工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则几千的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!
因此收集整理了一份《2024年Java开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。
既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Java开发知识点,真正体系化!
由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!
如果你觉得这些内容对你有帮助,可以扫码获取!!(备注Java获取)
读者福利
由于篇幅过长,就不展示所有面试题了,感兴趣的小伙伴
更多笔记分享
《互联网大厂面试真题解析、进阶开发核心学习笔记、全套讲解视频、实战项目源码讲义》点击传送门即可获取!
进阶开发核心学习笔记、全套讲解视频、实战项目源码讲义》点击传送门即可获取!**