深入解析JVM内部结构及GC机制的实战应用

一、JVM内部结构概述

JVM(jdk1.8)的内部结构主要包括以下几个部分:

  1. 类加载子系统(Class Loader Subsystem)
  2. 运行时数据区(Runtime Data Area)
  3. 执行引擎(Execution Engine)
  4. 本地方法接口(Native Interface)

1. 类加载子系统

负责加载和初始化Java类。它包括三个部分:加载器(ClassLoader)、校验器(Verifier)和解析器(Resolver)。

// 加载类
Class<?> clazz = Class.forName("com.example.MyClass");
// 创建实例
Object instance = clazz.newInstance();

2. 运行时数据区

运行时数据区是JVM内存的核心部分,用于存储程序执行时所需的数据。包含以下几个区域:

  • 方法区(Method Area):存储类信息、常量、静态变量、即时编译后的代码等。
  • 堆(Heap):用于存储对象实例,是GC的主要管理区域。
  • 栈(Stack):线程私有,存储局部变量、操作数栈、帧数据等。
  • 程序计数器(Program Counter Register):记录当前线程执行的字节码指令地址。
  • 本地方法栈(Native Method Stack):用于执行本地(Native)方法。
    // 堆上分配对象
    MyClass obj = new MyClass();
    // 方法区中存储类的字节码、常量池等信息

3. 执行引擎

执行引擎负责解释或编译字节码,并执行相应的机器指令。JVM执行引擎包含解释器(Interpreter)和即时编译器(JIT Compiler)。

// 解释器逐行解释执行字节码指令
int sum = 0;
for (int i = 1; i <= 10; i++) {sum += i;
}
System.out.println("Sum: " + sum);

4. 本地方法接口

本地方法接口(JNI,Java Native Interface)允许Java代码调用本地(C/C++)代码,实现与底层操作系统或其他编程语言的交互。

// 调用本地方法
System.loadLibrary("mylib");
nativeMethod();

二、JVM垃圾回收机制(GC)详解

JVM的垃圾回收机制是保障内存管理和系统性能的重要环节。主要的垃圾回收器包括:

  • Serial GC
  • Parallel GC
  • CMS GC
  • G1 GC

1. Serial GC

Serial GC是单线程的垃圾回收器,适用于单核CPU或内存较小的应用。其特点是简单高效,但在回收过程中会暂停所有应用线程(STW,Stop-The-World)。

2. Parallel GC

Parallel GC是多线程的垃圾回收器,能够利用多核CPU并行回收垃圾。适用于追求高吞吐量的应用,但仍然会发生STW。

3. CMS GC

CMS(Concurrent Mark-Sweep)GC是一种低延迟垃圾回收器,主要分为以下几个阶段:

  • 初始标记
  • 并发标记
  • 重新标记
  • 并发清除

CMS GC在大多数回收阶段可以与应用线程并发执行,减少了STW的时间。但其缺点是会产生内存碎片,并且在老年代无法回收时可能触发Full GC。

4. G1 GC

G1(Garbage First)GC是一种面向服务端应用的垃圾回收器,结合了CMS和Parallel GC的优点。G1 GC将堆内存划分为多个区域,并通过并发标记和回收算法,在指定的时间内优先回收垃圾较多的区域,确保低延迟和高吞吐量。

三、GC调优实战经验

1. 合理设置堆内存大小

通过监控应用的内存使用情况,合理设置堆内存的初始大小(-Xms)和最大大小(-Xmx),确保内存使用的稳定性和效率。

2. 选择合适的垃圾回收器

根据应用的特点和性能要求,选择合适的垃圾回收器。例如,对于低延迟要求的应用,可以选择CMS或G1 GC;对于高吞吐量要求的应用,可以选择Parallel GC。

3. 调整垃圾回收参数

根据实际情况,调整垃圾回收的相关参数,如:

1. 新生代和老年代比例调整
  • 参数: -XX:NewRatio
  • 说明: 该参数用于设置新生代与老年代的比例,默认值为2,表示新生代占整个堆的1/3。
  • 调整建议:
    • 如果应用中对象的生命周期较短,可以适当增加新生代的比例,减少老年代的空间,例如将比例调整为3或4。
    • 如果应用中对象的生命周期较长,可以适当降低新生代的比例,增加老年代的空间,例如将比例调整为1或1.5。
2. 新生代大小调整
  • 参数: -Xmn
  • 说明: 该参数用于设置新生代的初始大小。
  • 调整建议:
    • 如果新生代频繁发生Minor GC,可以尝试增大新生代的大小,以减少Minor GC的频率。
    • 如果新生代的对象存活率较低,可以适当减小新生代的大小,以减少内存的浪费。
3. Survivor区比例调整
  • 参数: -XX:SurvivorRatio
  • 说明: 该参数用于设置Eden区与Survivor区的比例,默认值为8,表示Eden区与每个Survivor区的比例为8:1。
  • 调整建议:
    • 如果应用中对象的存活率较高,可以适当增加Survivor区的比例,以增加对象在Survivor区的存活时间。
    • 如果应用中对象的存活率较低,可以适当减小Survivor区的比例,以减少空间的浪费。
4. CMS回收触发阈值调整
  • 参数: -XX:CMSInitiatingOccupancyFraction
  • 说明: 该参数用于设置老年代的使用率达到多少时触发CMS回收,默认值为68%。
  • 调整建议:
    • 如果老年代的使用率经常达到触发阈值,可以适当降低CMS回收的触发阈值,以增加CMS回收的频率。
    • 如果应用中CMS回收频繁导致性能问题,可以适当提高CMS回收的触发阈值,以减少CMS回收的频率。
5. 最大GC暂停时间调整
  • 参数: -XX:MaxGCPauseMillis
  • 说明: 该参数用于设置GC暂停的最大时间,默认值为不限制。
  • 调整建议:
    • 如果应用对GC暂停时间要求严格,可以设置一个较小的值,以限制GC暂停时间,例如设置为200ms。
    • 如果应用对GC暂停时间要求较为宽松,可以不设置该参数,以充分利用系统资源进行GC。
6. 并发GC线程数调整
  • 参数: -XX:ConcGCThreads
  • 说明: 该参数用于设置并发GC的线程数,默认值根据JVM的版本和配置而定。
  • 调整建议:
    • 如果应用中并发GC的线程数不足,可以适当增加该参数的值,以提高并发GC的效率。
    • 如果应用中并发GC的线程数过多,可以适当减小该参数的值,以减少系统资源的消耗。
7. 并行GC线程数调整
  • 参数: -XX:ParallelGCThreads
  • 说明: 该参数用于设置并行GC的线程数,默认值为CPU核心数的一半。
  • 调整建议:
    • 如果应用中并行GC的线程数不足,可以适当增加该参数的值,以提高并行GC的效率。
    • 如果应用中并行GC的线程数过多,可以适当减小该参数的值,以减少系统资源的消耗。

4. 使用GC日志进行分析

启用GC日志(-XX:+PrintGCDetails -Xloggc

.log),并使用工具(如GCViewer、GCEasy)对GC日志进行分析,找出性能瓶颈和内存泄漏点。

5. 避免频繁的Full GC

通过优化代码、减少对象创建和回收频率,避免频繁触发Full GC,从而提升系统性能和响应时间。

6. 实战调优示例及适用场景

示例1:高吞吐量应用的Parallel GC调优

适用场景:电商网站、支付系统等需要高并发处理请求的应用。

调优参数:

-XX:+UseParallelGC
-XX:ParallelGCThreads=8
-XX:NewRatio=2
-XX:SurvivorRatio=8
-XX:MaxGCPauseMillis=200
-XX:GCTimeRatio=4

解释:

  • UseParallelGC:启用Parallel GC。
  • ParallelGCThreads:设置并行GC的线程数为8。
  • NewRatio:设置新生代与老年代的比例为1:2。
  • SurvivorRatio:设置Eden区与Survivor区的比例为8:1。
  • MaxGCPauseMillis:设置最大GC暂停时间为200ms。
示例2:低延迟应用的CMS GC调优

适用场景:实时交易系统、金融应用等对响应时间要求较高的应用。

调优参数:

-XX:+UseConcMarkSweepGC
-XX:CMSInitiatingOccupancyFraction=75
-XX:+UseCMSInitiatingOccupancyOnly
-XX:CMSParallelRemarkEnabled=true
-XX:ConcGCThreads=4
-XX:ParallelGCThreads=8
-XX:+CMSClassUnloadingEnabled
-XX:+CMSScavengeBeforeRemark

解释:

  • UseConcMarkSweepGC:启用CMS GC。
  • CMSInitiatingOccupancyFraction:设置当老年代使用达到75%时触发CMS回收。
  • UseCMSInitiatingOccupancyOnly:仅在老年代使用率达到设定阈值时触发CMS回收。
  • CMSParallelRemarkEnabled:启用并行重新标记,减少STW时间。
  • ConcGCThreads:设置并发GC线程数为4。
示例3:大内存应用的G1 GC调优

适用场景:大数据处理、数据分析等需要大内存支持的应用。

调优参数:

-XX:+UseG1GC
-XX:MaxGCPauseMillis=200
-XX:InitiatingHeapOccupancyPercent=45
-XX:G1ReservePercent=10
-XX:ParallelGCThreads=8
-XX:ConcGCThreads=4
-XX:G1HeapRegionSize=32m
-XX:G1NewSizePercent=30
-XX:G1MaxNewSizePercent=60

解释:

  • UseG1GC:启用G1 GC。
  • MaxGCPauseMillis:设置最大GC暂停时间为200ms。
  • InitiatingHeapOccupancyPercent:设置在堆使用率达到45%时启动并发标记周期。
  • G1ReservePercent:保留堆内存的10%作为空闲区域,避免GC期间的内存不足。
  • ParallelGCThreads:设置并行GC的线程数为8。
  • ConcGCThreads:设置并发GC线程数为4。

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

React+TS前台项目实战(二)-- 路由配置 + 组件懒加载 + Error Boundary使用

文章目录 前言一、路由配置和懒加载lazy的使用二、TS版本Error Boundary组件封装三、在layout组件中使用Suspense组件和错误边界组件总结 前言 本文将详细介绍项目中的页面路由配置和异步组件懒加载处理&#xff0c;以提高用户体验&#xff0c;实现过渡效果。 一、路由配置和懒…

(2024,自监督 ViT,全监督 ViT,损失可视化,MAE,RC-MAE,自蒸馏,EMA)可视化自监督 ViT 的损失景观

Visualizing the loss landscape of Self-supervised Vision Transformer 公和众和号&#xff1a;EDPJ&#xff08;进 Q 交流群&#xff1a;922230617 或加 VX&#xff1a;CV_EDPJ 进 V 交流群&#xff09; 目录 0 摘要 2 基础&#xff1a;MAE 和 RC-MAE 3 损失景观 3.1 分…

QT C++(QT控件 QPushButton,QRadioButton,QCheckBox)

文章目录 1. QPushButton 普通按钮2. QRadioButton 单选按钮3. QCheckBox 复选按钮 1. QPushButton 普通按钮 QPushButton中的重要属性 text&#xff1a;按钮中的文本icon&#xff1a;按钮的图标iconSize&#xff1a;按钮中图标的尺寸shortCut&#xff1a;按钮对应的快捷键&a…

Unity3d使用3D WebView for Windows and macOS打开全景网页(720云)操作问题记录

问题描述 使用Unity3d内嵌网页的形式打开720云中的全景图这个功能&#xff0c;使用的是3D WebView for Windows and macOS插件&#xff0c;720云的全景图在浏览器上的操作是滑动鼠标滚轮推远/拉近全景图&#xff0c;鼠标左键拖拽网页可以旋转全景图内容。网页的打开过程是正常…

RK3588 Android13自定义一个按键实现长按短按

一、kernel修改 diff --git a/arch/arm64/boot/dts/rockchip/rk3588-nvr-demo.dtsi b/arch/arm64/boot/dts/rockchip/rk3588-nvr-demo.dtsi index 5aae5c613825..4cc1223f9cbf 100755 --- a/arch/arm64/boot/dts/rockchip/rk3588-nvr-demo.dtsib/arch/arm64/boot/dts/rockchip…

IDEA创建Mybatis项目

IDEA创建Mybatis项目 第一步&#xff1a;创建库表 -- 创建数据库 create database mybatis_db;-- 使用数据库 use mybatis_db;-- 创建user表 CREATE TABLE user (id INT AUTO_INCREMENT PRIMARY KEY,username VARCHAR(50) NOT NULL,password VARCHAR(50) NOT NULL,email VARC…

transformer中对于QKV的个人理解

目录 1、向量点乘 2、相似度计算举例 3、QKV分析 4、整体流程 (1) 首先从词向量到Q、K、V (2) 计算Q*&#xff08;K的转置&#xff09;&#xff0c;并归一化之后进行softmax (3) 使用刚得到的权重矩阵&#xff0c;与V相乘&#xff0c;计算加权求和。 5、多头注意力 上面…

记一次postgresql拼接函数string_agg() 和row_number() 使用

PG两个函数使用需求和简单介绍 需求背景介绍第一个需求背景是这样的需求升级一下接下来讲讲STRING_AGG()基本语法排序 然后我们再说说ROW_NUMBER()基本语法使用 row_number() over (partition by) 进行分组统计使用 row_num限定每组数量 需求背景介绍 第一个需求背景是这样的 …

【MATLAB源码-第222期】基于matlab的改进蚁群算法三维栅格地图路径规划,加入精英蚁群策略。包括起点终点,障碍物,着火点,楼梯。

操作环境&#xff1a; MATLAB 2022a 1、算法描述 蚁群算法&#xff08;Ant Colony Optimization&#xff0c;ACO&#xff09;是一种通过模拟蚂蚁觅食行为的启发式优化算法。它由意大利学者Marco Dorigo在20世纪90年代初提出&#xff0c;最初用于解决旅行商问题&#xff08;T…

从《千脑智能》看大模型

千脑智能与大模型 千脑智能介绍 世界模型千脑智能理论——对大脑的全新理解旧大脑&#xff1a;演化的历史烙印新大脑&#xff1a;智慧的创新引擎新旧大脑的互动与争斗启示与借鉴 大脑对信息的处理和建模六根六尘六识 新脑&#xff1a;智能的创新中枢旧脑&#xff1a;生存的本能…

Web前端笔记:深入探索与实战精髓

Web前端笔记&#xff1a;深入探索与实战精髓 Web前端&#xff0c;作为构建互联网世界的基石之一&#xff0c;承载了用户与网页交互的桥梁作用。对于前端开发者而言&#xff0c;不断积累与更新自己的知识体系显得尤为重要。本文将从四个方面、五个方面、六个方面和七个方面&…

Spring的Controller是单例还是多例,如何保证线程安全的。

目录 验证是否单例&#xff08;默认单例&#xff09; 多例测试 单例对象成员变量测试 多例对象成员变量测试 解决方案 结论&#xff1a; 补充说明 答案&#xff1a;controller默认是单例的&#xff0c;不要使用非静态的成员变量&#xff0c;否则会发生数据逻辑混乱。 正…

求宇文玥在水下的浮力和赵丽颖捞他的时间

关注微信公众号 数据分析螺丝钉 免费领取价值万元的python/java/商业分析/数据结构与算法学习资料 2024年汉东省在达康书记的带领下率先实现高考试点改革。为让更多的考生能提升对他们的理解和记忆&#xff0c;把电视剧的场景融入考试题目中。确保学生看一遍就懂&#xff0c;想…

STM32 proteus + STM32Cubemx仿真教程(第一课LED教程)

文章目录 前言一、STM32点亮LED灯的原理1.1GPIO是什么1.2点亮LED灯的原理 二、STM32Cubemx创建工程三、proteus仿真电路图四、程序代码编写1.LED灯操作函数介绍HAL_GPIO_WritePin函数原型参数说明示例代码 HAL_GPIO_TogglePin函数原型参数说明示例代码 2.代码编写3.烧写程序 总…

(三)React事件

1. React基础事件绑定 语法&#xff1a; on 事件名称 { 事件处理程序 }&#xff0c;整体上遵循驼峰命名法 App.js //项目根组件 //App -> index.js -> public/index.html(root)function App() {const handleClick () > {console.log(button被点击了)}return (<…

【值得一看的新特性】JDK21

文章目录 1. JEP 425: 虚拟线程 (Virtual Threads)1.1 是什么1.2 为什么1.3 怎么用 2. JEP 428: 结构化并发 (Structured Concurrency)2.1 是什么2.2 为什么2.3 怎么用 3. JEP 440: Record模式 (Record Patterns)3.1 是什么Record类型 3.2 为什么3.3 怎么用 4. JEP 427: switch…

9.0 Android中的网络技术

Android中网络相关的技术&#xff0c;主要分别两种&#xff0c;一种为直接显示网页&#xff0c;另外一种为获取服务器中的数据进行设置。 权限声明 访问网络是需要声明权限 <manifest xmlns:android"http://schemas.android.com/apk/res/android"package"…

mac m1使用docker安装mysql5.7,并且开启binlog

1. 使用 mysql/mysql-server:5.7 镜像 创建一个名为 docker-compose.yml 的文件&#xff0c;内容如下&#xff1a; version: 3.1services:mysql:image: mysql/mysql-server:5.7container_name: mysql57ports:- "3306:3306"environment:MYSQL_ROOT_PASSWORD: yourpa…

k8s学习--kubernetes服务自动伸缩之水平收缩(pod副本收缩)VPA详细解释与安装

文章目录 前言VPA简介简单理解详细解释VPA的优缺点优点1.自动化资源管理2.资源优化3.性能和稳定性提升5.成本节约6.集成性和灵活性 缺点1.Pod 重启影响可用性2.与 HPA 冲突3.资源监控和推荐滞后&#xff1a;4.实现复杂度&#xff1a; 核心概念Resource Requests 和 Limits自动调…

51单片机采用定时器T1的方式1的中断计数方式,外接开关K4按4次后,8只LED闪烁不停

1、功能描述 采用定时器T1的方式1的中断计数方式&#xff0c;外接开关K4按4次后&#xff0c;8只LED闪烁不停 2、实验原理 定时器原理:8051的定时器可以用于计数外部事件或执行内部定时操作。在本程序中&#xff0c;定时器1被设置为模式2&#xff0c;即8位自动重装载定时器模式…