SpringBoot极简教程 · Spring Boot 禅与计算机程序设计艺术

【图文详解系列】JVM 内存模型

2020-05-22  本文已影响0人  光剑书架上的书

内存模型

(1)线程私有区:

(2)线程共享区:

进程计数器PC

进程计数器PC,当前线程所执行的字节码行号指示器。每个线程都有自己计数器,是私有内存空间,该区域是整个内存中较小的一块。

当线程正在执行一个Java方法时,PC计数器记录的是正在执行的虚拟机字节码的地址;当线程正在执行的一个Native方法时,PC计数器则为空(Undefined)。

虚拟机栈

虚拟机栈,生命周期与线程相同,是Java方法执行的内存模型。每个方法(不包含native方法)执行的同时都会创建一个栈帧结构,方法执行过程,对应着虚拟机栈的入栈到出栈的过程。

栈帧(Stack Frame)结构

栈帧是用于支持虚拟机进行方法执行的数据结构,是属性运行时数据区的虚拟机站的栈元素。见上图, 栈帧包括:

  1. 局部变量表 (locals大小,编译期确定),一组变量存储空间, 容量以slot为最小单位。
  2. 操作栈(stack大小,编译期确定),操作栈元素的数据类型必须与字节码命令序列严格匹配
  3. 动态连接, 指向运行时常量池中该栈帧所属方法的引用,为了 动态连接使用。
    • 前面的解析过程其实是静态解析;
    • 对于运行期转化为直接引用,称为动态解析。
  4. 方法返回地址
    • 正常退出,执行引擎遇到方法返回的字节码,将返回值传递给调用者
    • 异常退出,遇到Exception,并且方法未捕捉异常,那么不会有任何返回值。
  5. 额外附加信息,虚拟机规范没有明确规定,由具体虚拟机实现。

Java堆

Java堆,是Java虚拟机管理的最大的一块内存,也是GC的主战场,里面存放的是几乎所有的对象实例和数组数据。JIT编译器有栈上分配、标量替换等优化技术的实现导致部分对象实例数据不存在Java堆,而是栈内存。

对象创建的过程是在堆上分配着实例对象,那么对象实例的具体结构如下:

对于填充数据不是一定存在的,仅仅是为了字节对齐。HotSpot VM的自动内存管理要求对象起始地址必须是8字节的整数倍。对象头本身是8的倍数,当对象的实例数据不是8的倍数,便需要填充数据来保证8字节的对齐。该功能类似于高速缓存行的对齐。

另外,关于在堆上内存分配是并发进行的,虚拟机采用CAS加失败重试保证原子操作,或者是采用每个线程预先分配TLAB内存.

对象分配规则

参考资料

  1. http://gityuan.com/2016/01/09/java-memory/
  2. http://blog.csdn.net/universe_ant/article/details/58585854
  3. Java —— 运行时栈帧结构
  4. https://www.dazhuanlan.com/2019/12/07/5deaf86336d1d/

Kotlin开发者社区

专注分享 Java、 Kotlin、Spring/Spring Boot、MySQL、redis、neo4j、NoSQL、Android、JavaScript、React、Node、函数式编程、编程思想、"高可用,高性能,高实时"大型分布式系统架构设计主题。

High availability, high performance, high real-time large-scale distributed system architecture design

分布式框架:Zookeeper、分布式中间件框架等
分布式存储:GridFS、FastDFS、TFS、MemCache、redis等
分布式数据库:Cobar、tddl、Amoeba、Mycat
云计算、大数据、AI算法
虚拟化、云原生技术
分布式计算框架:MapReduce、Hadoop、Storm、Flink等
分布式通信机制:Dubbo、RPC调用、共享远程数据、消息队列等
消息队列MQ:Kafka、MetaQ,RocketMQ
怎样打造高可用系统:基于硬件、软件中间件、系统架构等一些典型方案的实现:HAProxy、基于Corosync+Pacemaker的高可用集群套件中间件系统
Mycat架构分布式演进
大数据Join背后的难题:数据、网络、内存和计算能力的矛盾和调和
Java分布式系统中的高性能难题:AIO,NIO,Netty还是自己开发框架?
高性能事件派发机制:线程池模型、Disruptor模型等等。。。

合抱之木,生于毫末;九层之台,起于垒土;千里之行,始于足下。不积跬步,无以至千里;不积小流,无以成江河。

上一篇 下一篇

猜你喜欢

热点阅读