《JVM由浅入深学习【四】 2023-12-24》JVM由简入深学习提升分享

2024-01-03 04:52

本文主要是介绍《JVM由浅入深学习【四】 2023-12-24》JVM由简入深学习提升分享,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

JVM由简入深学习提升分享四

  • 1.JVM中java堆的特点及作用
  • 2. JVM中对象如何在堆内存中分配
  • 3. JVM堆内存中的对象布局

1.JVM中java堆的特点及作用

  • 是线程共享的一块区域
  • 虚拟机启动时就创建了
  • 是虚拟机中内存占用很大的一块
  • 存放所有的实例对象和数组
  • GC主要的作用区域
  • 可分为新生代(刚创建)和老年代(存活很久)
  • 新生代更细化可分为Eden,From Survivor,To Survivor,比例为8:1:1
    jvm堆结构图
  • 可以通过-Xms,-Xmx调节堆大小
  • 如果从分配内存的角度看,所有线程共享的 Java 堆中可以划分出多个线程私有的分配缓冲区 (Thread Local Allocation Buffer,TLAB) ,以提升对象分配时的效率
  • 无法在拓展:java.lang.OutOfMemoryError: Java heap space
    例子:模拟OOM,设置JVM参数-Xmx128m -Xms128m,这里最大最小都设置128M,我的Customer里面有个bytes变量,一个就有1m,所以每个CUstomer都多于1M,所以运行程序customerList长度到达一定次数就OOM了
    在这里插入图片描述
@Data
public class Customer {private int no;private String username;private BigDecimal money;private byte[] a = new byte[1024 * 1024]; // 这里1024*1024byte等于1m
}
public class MyTestOOM {public static void main(String[] args) {List<Customer> customerList = new ArrayList<>();while (true) {Customer customer = new Customer();customer.setNo(1);customer.setUsername("testOOM");customer.setMoney(new BigDecimal("1000"));customerList.add(customer);System.out.println(customerList.size());}}
}

在这里插入图片描述

2. JVM中对象如何在堆内存中分配

  1. 指针碰撞(Bump The Poniter):内存规整的情况下
  2. 空闲列表(Free List):内存不规整的情况下
    以上两种方式要看垃圾回收器是否有空间压缩整理的能力来决定
  • 例如Serial,ParNew有有带压缩整理能力的收集器,则采用的是指针碰撞,简单高效
  • 例如CMS这种基于清除(Sweep)算法的收集器时,理论上采用较为复杂的空间列表分配方式
  1. 本地线程分配缓冲(TLAB):对象创建在虚拟机中频繁发生,即使仅仅修改一个指针所指向的位置,在并发情况下也并不是线程安全的,可能出现正在给对象 A分配内存,指针还没来得及修改,对象 B 又同时使用了原来的指针来分配内存的情况;
    两种解决方案
  • 同步锁定:JVM 是采用 CAS 配上失败重试的方式保证更新操作的原予性,
  • 线程锁定:把内存分配的动作按照线程划分在不同的空间之中进行,即每个线程在 Java堆中预先分配一小块内存,称为本地线程分配缓冲(Thread Local Allocation Buffer,TLAB),哪个线程要分配内存,就在哪个线程的本地缓冲区中分配,只有本地缓冲区用完了,分配新的缓存区时才需要同步锁定,虚拟机是否使用TLAB,可以通过-XX: +/-UseTLAB 参数来设定,-XX:TLABSize=512k设置大小
    在这里插入图片描述
    在这里插入图片描述
    java -XX:+PrintFlagsFinal来打印信息,可找到TLAB,默认开启,自动分配容量
    在这里插入图片描述

3. JVM堆内存中的对象布局

  • 在 HotSpot 虚拟机中,一个对象的存储结构分为 3 块区域: 对象头(Header)、:实例数据(Instance Data) 和 对齐填充(Padding)
  • 对象头(Header): 包含两部分,
    • 第一部分用于存储对象自身的运行时数据,如哈希码、GC分代年龄、锁状态标志、线程持有的锁、偏向线程 ID、偏向时间戳等,32 位虚拟机占 32 bit.64 位虚拟机占 64 bit,官方称为 “Mark Word’
    • 第二部分是类型指针,即对象指向它的类的元数据指针,虚拟机通过这个指针确定这个对象是哪个类的实例,另外,如果是Java 数组,对象头中还必须有一块用于记录数组长度的数据因为普通对象可以通过 Java 对象元数据确定大小,而数组对象不可以;
  • 实例数据(Instance Data): 程代码中所定义的各种成员变量类型的字段内容(包含父类继承下来的和子类中定义的);
  • 对齐填充(Padding): 不是必然需要,主要是占位,保证对象大小是某个字节的整数倍,HotSpot 虚拟机,任何对象的大小都是 8 字节的整数倍。如果不是整数倍就靠这个填充

这篇关于《JVM由浅入深学习【四】 2023-12-24》JVM由简入深学习提升分享的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!


原文地址:
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.chinasem.cn/article/564613

相关文章

Java中JSON格式反序列化为Map且保证存取顺序一致的问题

《Java中JSON格式反序列化为Map且保证存取顺序一致的问题》:本文主要介绍Java中JSON格式反序列化为Map且保证存取顺序一致的问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未... 目录背景问题解决方法总结背景做项目涉及两个微服务之间传数据时,需要提供方将Map类型的数据序列化为co

Java Lambda表达式的使用详解

《JavaLambda表达式的使用详解》:本文主要介绍JavaLambda表达式的使用方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、前言二、Lambda表达式概述1. 什么是Lambda表达式?三、Lambda表达式的语法规则1. 无参数的Lambda表

java中Optional的核心用法和最佳实践

《java中Optional的核心用法和最佳实践》Java8中Optional用于处理可能为null的值,减少空指针异常,:本文主要介绍java中Optional核心用法和最佳实践的相关资料,文中... 目录前言1. 创建 Optional 对象1.1 常规创建方式2. 访问 Optional 中的值2.1

Spring Boot 整合 Apache Flink 的详细过程

《SpringBoot整合ApacheFlink的详细过程》ApacheFlink是一个高性能的分布式流处理框架,而SpringBoot提供了快速构建企业级应用的能力,下面给大家介绍Spri... 目录Spring Boot 整合 Apache Flink 教程一、背景与目标二、环境准备三、创建项目 & 添

Spring组件实例化扩展点之InstantiationAwareBeanPostProcessor使用场景解析

《Spring组件实例化扩展点之InstantiationAwareBeanPostProcessor使用场景解析》InstantiationAwareBeanPostProcessor是Spring... 目录一、什么是InstantiationAwareBeanPostProcessor?二、核心方法解

深入解析 Java Future 类及代码示例

《深入解析JavaFuture类及代码示例》JavaFuture是java.util.concurrent包中用于表示异步计算结果的核心接口,下面给大家介绍JavaFuture类及实例代码,感兴... 目录一、Future 类概述二、核心工作机制代码示例执行流程2. 状态机模型3. 核心方法解析行为总结:三

Spring @RequestMapping 注解及使用技巧详解

《Spring@RequestMapping注解及使用技巧详解》@RequestMapping是SpringMVC中定义请求映射规则的核心注解,用于将HTTP请求映射到Controller处理方法... 目录一、核心作用二、关键参数说明三、快捷组合注解四、动态路径参数(@PathVariable)五、匹配请

Java -jar命令如何运行外部依赖JAR包

《Java-jar命令如何运行外部依赖JAR包》在Java应用部署中,java-jar命令是启动可执行JAR包的标准方式,但当应用需要依赖外部JAR文件时,直接使用java-jar会面临类加载困... 目录引言:外部依赖JAR的必要性一、问题本质:类加载机制的限制1. Java -jar的默认行为2. 类加

Java进程CPU使用率过高排查步骤详细讲解

《Java进程CPU使用率过高排查步骤详细讲解》:本文主要介绍Java进程CPU使用率过高排查的相关资料,针对Java进程CPU使用率高的问题,我们可以遵循以下步骤进行排查和优化,文中通过代码介绍... 目录前言一、初步定位问题1.1 确认进程状态1.2 确定Java进程ID1.3 快速生成线程堆栈二、分析

Swagger在java中的运用及常见问题解决

《Swagger在java中的运用及常见问题解决》Swagger插件是一款深受Java开发者喜爱的工具,它在前后端分离的开发模式下发挥着重要作用,:本文主要介绍Swagger在java中的运用及常... 目录前言1. Swagger 的主要功能1.1 交互式 API 文档1.2 客户端 SDK 生成1.3