想要学会JVM调优,先掌握JVM内存模型和JVM运行原理

news2025/4/23 4:34:34

1、前言

今天将和你一起探讨Java虚拟机(JVM)的性能调优。

JVM算是面试中的高频问题了,通常情况下总会有人问到:请你讲解下 JVM 的内存模型,JVM 的

性能调优做过

2、为什么 JVM 在 Java 中如此重要

首先你应该知道,运行一个Java应用程序,我们必须要先安装 JDK 或者 JRE 包。

这是因为 Java应用在编译后会变成字节码,然后通过字节码运行在 JVM 中,而 JVM 是 JRE 的核

心组成部分。

JVM不仅承担了Java字节码的分析(JIT compiler)和执行(Runtime),同时也内置了自动内

存分配管理机制

这个机制可以大大降低手动分配回收机制可能带来的内存泄露和内存溢出风险,使 Java开发人员

不需要关注每个对象的内存分配以及回收,从而更专注于业务本身。

3、从了解内存模型开始

JVM 自动内存分配管理机制的好处很多,但实则是把双刃剑

这个机制在提升Java开发效率的同时,也容易使 Java 开发人员过度依赖于自动化,弱化对内存的

管理能力,这样系统就很容易发生 JVM 的堆内存异常,垃圾回收(GC)的方式不合适以及 GC

次数过于频繁等问题,这些都将直接影响到应用服务的性能

因此,要进行 JVM 层面的调优,就需要深入了解 JVM 内存分配和回收原理,这样在遇到问题

时,我们才能通过日志分析快速地定位问题;也能在系统遇到性能瓶颈时,通过分析JVM 调优来

优化系统性能

这也是整个模块的重点内容,今天我们就从 JVM 的内存模型学起,为后续的学习打下一个坚实的

基础。

4、JVM内存模型的具体设计

我们先通过一张 JVM 内存模型图,来熟悉下其具体设计。

在 Java 中,JVM内存模型主要分为堆、程序计数器、方法区、虚拟机栈和本地方法栈

我们来分析,JVM 的 5 个分区具体是怎么实现的呢?

1.堆(Heap)

堆是 JVM 内存中最大的一块内存空间,该内存被所有线程共享,几乎所有对象和数组都被分配到

了堆内存中。

堆被划分为新生代和老年代,新生代又被进一步划分为 Eden 和Survivor 区,最后 Survivor 由

From Survivor和To Survivor 组成。

Java6 版本中,永久代在非堆内存区;到了 Java7 版本,永久代的静态变量和运行时常量池

合并到了堆中;而到了 Java8永久代被元空间取代了

结构如下图所示:

2.程序计数器(Program Counter Register)

程序计数器是一块很小的内存空间,主要用来记录各个线程执行的字节码的地址,例如,分支、循

环、跳转、异常、线程恢复等都依赖于计数器。

由于Java是多线程语言,当执行的线程数量超过CPU数量时,线程之间会根据时间片轮询争夺

CPU资源。

如果一个线程的时间片用完了,或者是其它原因导致这个线程的CPU资源被提前抢夺,那么这个退

出的线程就需要单独的一个程序计数器,来记录下一条运行的指令。

3.方法区(Method Area)

很多开发者都习惯将方法区称为“永久代”,其实这两者并不是等价的。

HotSpot 虚拟机使用永久代来实现方法区,但在其它虚拟机中,例如,Oracle的JRockit、IBM 的

J9 就不存在永久代一说。

因此,方法区只是 JVM 中规范的一部分,可以说,在HotSpot 虚拟机中,设计人员使用了永久代

来实现了 JVM 规范的方法区。

方法区主要是用来存放已被虚拟机加载的类相关信息,包括类信息、运行时常量池、字符串常量

。类信息又包括了类的版本、字段、方法、接口和父类等信息。

JVM在执行某个类的时候,必须经过加载、连接、初始化,而连接又包括验证、准备、解析三个阶

段。

在加载类的时候,JVM会先加载 class 文件,而在 class 文件中除了有类的版本、字段、方法和接

口等描述信息外,还有一项信息是常量池 (Constant Pool Table),用于存放编译期间生成的各种字

面量和符号引用。

字面量包括字符串(String a=“b”)、基本类型的常量(final 修饰的变量),符号引用则包括类和

方法的全限定名(例如 String 这个类,它的全限定名就是Java/lang/String)、字段的名称描述

以及方法的名称描述符

而当类加载到内存中后,JVM 就会将 class 文件常量池中的内容存放到运行时的常量池中;在解析

阶段,JVM 会把符号引用替换为直接引用(对象的索引值)。

例如,类中的一个字符串常量在 class 文件中时,存放在 class 文件常量池中的;在 JVM加载完类

之后,JVM 会将这个字符串常量放到运行时常量池中,并在解析阶段,指定该字符串对象的索引

值。

运行时常量池是全局共享的,多个类共用一个运行时常量池,class 文件中常量池多个相同的字符

串在运行时常量池只会存在一份。

方法区与堆空间类似,也是一个共享内存区,所以方法区是线程共享的。假如两个线程都试图访问

方法区中的同一个类信息,而这个类还没有装入 JVM,那么此时就只允许一个线程去加载它,另

一个线程必须等待。

HotSpot 虚拟机Java7 版本中已经将永久代的静态变量和运行时常量池转移到了堆中其余

部分则存储在JVM 的非堆内存中,而 Java8 版本已经将方法区中实现的永久代去掉了,并用元空

间(class metadata)代替了之前的永久代,并且元空间的存储位置是本地内存。

之前永久代的类的元数据存储在了元空间,永久代的静态变量(class static variables)以及运行

时常量池(runtime constant pool)则跟 Java7 一样,转移到了堆中。

那你可能又有疑问了,Java8 为什么使用元空间替代永久代,这样做有什么好处呢?

官方给出的解释是:

移除永久代是为了融合 HotSpot JVM 与 JRockit VM 而做出的努力,因为JRockit 没有永久代,所

以不需要配置永久代。

永久代内存经常不够用或发生内存溢出,爆出异常 java.lang.OutOfMemoryError: PermGen。

这是因为在 JDK1.7 版本中,指定的 PermGen 区大小为 8M,由于PermGen 中类的元数据信息在

每次 FullGC 的时候都可能被收集,回收率都偏低,成绩很难令人满意;还有为 PermGen分配多

大的空间很难确定PermSize的大小依赖于很多因素,比如,JVM加载的 class 总数、常量池的大

小和方法的大小等。

4.虚拟机栈(VM stack)

Java虚拟机栈是线程私有的内存空间,它和Java 线程一起创建

当创建一个线程时,会在虚拟机栈中申请一个线程栈,用来保存方法的局部变量、操作数栈、动态

链接方法和返回地址等信息,并参与方法的调用和返回。每一个方法的调用都伴随着栈帧的入栈操

作,方法的返回则是栈帧的出栈操作。

5.本地方法栈(Native Method Stack)

本地方法栈跟 Java 虚拟机栈的功能类似,Java虚拟机栈用于管理 Java 函数的调用,而本地方法

栈则用于管理本地方法的调用。

但本地方法并不是用 Java 实现的,而是由C语言实现的。

5、JVM的运行原理

看到这里,相信你对 JVM 内存模型已经有个充分的了解了。

接下来,我们通过一个案例来了解下代码和对象是如何分配存储的,Java 代码又是如何在 JVM 中运行的。

 
public class JVMCase {
     // 常量
     public final static String MAN_SEX_TYPE = "man";
     // 静态变量
     public static String WOMAN_SEX_TYPE = "woman";
     
     public static void main(String[] args) {
         Student stu = new Student();
         stu.setName("nick");
         stu.setSexType(MAN_SEX_TYPE);
         stu.setAge(20);
         JVMCase jvmcase = new JVMCase();
         // 调用静态方法
         print(stu);
         // 调用非静态方法
         jvmcase.sayHello(stu);
     }
     // 常规静态方法
     public static void print(Student stu) {
         System.out.println("name: " + stu.getName() + "; sex:" + stu.getSexType
     }
     // 非静态方法
     public void sayHello(Student stu) {
         System.out.println(stu.getName() + "say: hello"); 
     }
 }
                            
 class Student{
     private String name;
     private String sexType;
     private int age;
     
     public String getName() {
     return name;
     }
     public void setName(String name) {
     this.name = name;
     }
     public String getSexType() {
     return sexType;
     }
     public void setSexType(String sexType) {
     this.sexType = sexType;
     }
     public int getAge() {
     return age;
     }
     public void setAge(int age) {
     this.age = age;
     }
 }

当我们通过 Java 运行以上代码时,JVM 的整个处理过程如下:

1.JVM向操作系统申请内存

JVM 第一步就是通过配置参数或者默认配置参数向操作系统申请内存空间,根据内存大小找到具

体的内存分配表,然后把内存段的起始地址和终止地址分配给 JVM,接下来 JVM 就进行内部分

配。

2.JVM进行内部分配

JVM 获得内存空间后,会根据配置参数分配堆、栈以及方法区的内存大小

3.class文件加载、验证、准备以及解析

其中准备阶段会为类的静态变量分配内存,初始化为系统的初始值。

4.初始化阶段

完成上一个步骤后,将会进行最后一个初始化阶段。

在这个阶段中,JVM首先会执行构造器 <clinit> 方法编译器会在.java 文件被编译成.class 文件

,收集所有类的初始化代码,包括静态变量赋值语句、静态代码块、静态方法,收集在一起成为

<clinit>() 方法。

5.执行方法

启动main线程,执行main方法,开始执行第一行代码。

此时堆内存中会创建一个student对象,对象引用 student 就存放在栈中

此时再次创建一个 JVMCase 对象,调用 sayHello 非静态方法,sayHello 方法属于对象

JVMCase,此时sayHello 方法入栈,并通过栈中的 student 引用调用堆中的 Student对象;之后,

调用静态方法 print,print静态方法属于 JVMCase 类,是从静态方法中获取,之后放入到栈中,也

是通过 student 引用调用堆中的student 对象。

了解完实际代码在 JVM 中分配的内存空间以及运行原理,相信你会更加清楚内存模型中各个区域

的职责分工。

6、总结

我们主要学习了最基础的内存模型设计了解其各个分区的作用及实现原理

在 Java 中,JVM内存模型主要分为堆、程序计数器、方法区、虚拟机栈和本地方法栈

  • 堆是 JVM 内存中最大的一块内存空间,该内存被所有线程共享

  • 程序计数器是一块很小的内存空间,主要用来记录各个线程执行的字节码的地址,例如,支、循环、跳转、异常、线程恢复等都依赖于计数器。

  • 方法区只是 JVM 中规范的一部分,方法区主要是用来存放已被虚拟机加载的类相关信息,包括类信息、运行时常量池、字符串常量池

  • Java虚拟机栈用于管理 Java 函数的调用,而本地方法栈则用于管理本地方法的调用。

如今,JVM在很大程度上减轻了Java开发人员投入到对象生命周期的管理精力。

在使用对象的时候,JVM会自动分配内存给对象,在不使用的时候,垃圾回收器会自动回收对象,

释放占用的内存。

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

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

相关文章

近 300 个假冒应用程序泛滥成灾,淹没伊朗银行业

内容概述&#xff1a; 近期&#xff0c;针对伊朗银行业的大规模活动规模不断扩大&#xff0c;近 300 个恶意 Android 应用程序针对用户的账户凭据、信用卡和加密钱包发起攻击。四个月前&#xff0c;Sophos 的研究人员详细介绍了一场漫长的活动&#xff0c;涉及 40 个恶意银行应…

【docker】—— Docker 简介

目录 &#xff08;一&#xff09;容器技术发展史 1、Jail 时代 2、云时代 3、云原生时代 &#xff08;二&#xff09;编排与容器的技术演进之路 1、DockerClient 2、RUNC&Shim 3、CRI-Containerd 4、CRI-O 5、Containerd &#xff08;三&#xff09;Docker 简介…

【开源】基于Vue+SpringBoot的二手车交易系统

目录 一、摘要1.1 项目介绍1.2 项目录屏 二、功能模块2.1 数据中心模块2.2 二手车档案管理模块2.3 车辆预约管理模块2.4 车辆预定管理模块2.5 车辆留言板管理模块2.6 车辆资讯管理模块 三、系统设计3.1 E-R图设计3.2 可行性分析3.2.1 技术可行性分析3.2.2 操作可行性3.2.3 经济…

苹果紧急修复两大零日漏洞,影响iPhone、iPad 和 Mac 设备

内容概述&#xff1a; 近日&#xff0c;苹果公司发布紧急安全更新&#xff0c;此次更新修复了两个在攻击中被利用并影响 iPhone、iPad 和 Mac 设备的零日漏洞。据统计&#xff0c;自今年年初以来已修复的零日漏洞数量已达到 20 个。其中提到此次发现的零日漏洞很可能已被iOS 1…

ERD Online更换Licence为最友好的MIT协议

ERD Online一直秉承着开放、灵活、用户友好的理念&#xff0c;为用户提供高品质的服务。我们非常激动地宣布&#xff0c;ERD Online的许可证已经进行了重大更新&#xff0c;将采用MIT&#xff08;麻省理工学院&#xff09;协议&#xff0c;这一变更旨在进一步提升用户体验&…

大模型LLM的微调技术:LoRA

0 引言 LoRA(Low-Rank Adaptation)出自2021年的论文“LoRA: Low-Rank Adaptation of Large Language Models” LoRA技术冻结预训练模型的权重&#xff0c;并在每个Transformer块中注入可训练层&#xff08;称为秩分解矩阵&#xff09;&#xff0c;即在模型的Linear层的旁边增…

nextTick的原理

开发中有这么一个需求&#xff0c;回显的适合&#xff0c;el-tree的检查严格标志属性更新为true。当更新完成后&#xff0c;又要改为false。还原。 <template><div><el-tree:data"data"show-checkbox:check-strictly"checkStrictly"default…

[LLM]大模型训练(三)--DeepSpeed-Train

安装DeepSpeed与集成 pip install deepspeed DeepSpeed与HuggingFace Transformers直接集成。使用者可以通过在模型训练命令中加入简单的 --deepspeed 标志和配置文件&#xff0c;来轻松加速模型训练。 编写DeepSpeed模型 DeepSpeed模型训练的核心是什么&#xff1f;它如何处…

极智开发 | 解读英伟达软件生态 深度神经网络库cuDNN

欢迎关注,获取我的更多经验分享 大家好,我是极智视界,本文来介绍一下 解读英伟达软件生态 深度神经网络库cuDNN。 邀您加入我的知识星球「极智视界」,星球内有超多好玩的项目实战源码下载,链接:https://t.zsxq.com/0aiNxERDq cuDNN,全称为 NVIDIA CUDA Deep Neural Net…

[设计模式 Go实现] 创建型~抽象工厂模式

抽象工厂模式用于生成产品族的工厂&#xff0c;所生成的对象是有关联的。 如果抽象工厂退化成生成的对象无关联则成为工厂函数模式。 比如本例子中使用RDB和XML存储订单信息&#xff0c;抽象工厂分别能生成相关的主订单信息和订单详情信息。 如果业务逻辑中需要替换使用的时候…

Spring Data Redis对象缓存序列化问题

相信在项目中&#xff0c;你一定是经常使用 Redis &#xff0c;那么&#xff0c;你是怎么使用的呢&#xff1f;在使用时&#xff0c;有没有遇到同我一样&#xff0c;对象缓存序列化问题的呢&#xff1f;那么&#xff0c;你又是如何解决的呢&#xff1f; Redis 使用示例 添加依…

nodejs微信小程序+python+PHP的冷链物流配送系统-计算机毕业设计推荐

目 录 摘 要 I ABSTRACT II 目 录 II 第1章 绪论 1 1.1背景及意义 1 1.2 国内外研究概况 1 1.3 研究的内容 1 第2章 相关技术 3 2.1 nodejs简介 4 2.2 express框架介绍 6 2.4 MySQL数据库 4 第3章 系统分析 5 3.1 需求分析 5 3.2 系统可行性分析 5 3.2.1技术可行性&#xff1a;…

【嵌入式开发 Linux 常用命令系列 7.3 -- linux 命令行数值计算】

文章目录 linux 命令行数值计算使用 awk使用 bc 命令使用 Bash 的内置算术扩展使用 expr脚本命令实现 linux 命令行数值计算 在 Linux 命令行中&#xff0c;您可以使用多种方法来执行基本的数学运算。以下是一些示例&#xff1a; 使用 awk awk 是一个强大的文本处理工具&…

如何优化旋转花键的装配方式?

花键轴与花键套的装配在工业上广泛应用&#xff0c;装配的质量受花键轴与花键套间的接触状态、对应的受力情况及相对位置关系影响&#xff0c;那么&#xff0c;我们应该如何优化旋转花键的装配方式呢&#xff1f; 确保轴和孔的配合精度是关键&#xff0c;可以采用高精度的加工和…

基于Python的电商手机数据可视化分析和推荐系统

1. 项目简介 本项目旨在通过Python技术栈对京东平台上的手机数据进行抓取、分析并构建一个简单的手机推荐系统。主要功能包括&#xff1a; 网络爬虫&#xff1a;从京东获取手机数据&#xff1b;数据分析&#xff1a;统计各厂商手机销售分布、市场占有率、价格区间和好评率&am…

15.备份与恢复

目录 1、数据备份 1、使用MySQLdump 命令备份 2、使用MySQLdump备份数据库中的某个表 3、使用MySQLdump备份多个数据库 4、直接复制整个数据库目录 2、数据恢复 1 、使用MySQL命令恢复 2、直接复制到数据库目录 3、表的导入导出 1、 使用SELECT ... INTO OUTFILE导出…

PHP序列化总结3--反序列化的简单利用及案例分析

反序列化中生成对象里面的值&#xff0c;是由反序列化里面的值决定&#xff0c;与原类中预定义的值的值无关&#xff0c;穷反序列化的对象可以使用类中的变量和方法 案例分析 反序列化中的值可以覆盖原类中的值 我们创建一个对象&#xff0c;对象创建的时候触发了construct方…

企业计算机服务器中了360后缀勒索病毒如何处理,勒索病毒应对步骤

网络技术的应用与发展&#xff0c;为企业的生产运营提供了有力保障&#xff0c;但也为网络安全威胁埋下隐患。近期&#xff0c;网络上的勒索病毒非常嚣张&#xff0c;严重影响了企业的生产运营。近日&#xff0c;云天数据恢复中心接到很多企业的求助&#xff0c;企业的计算机服…

javascript 常见事件

简介&#xff1a; JavaScript&#xff08;简称“JS”&#xff09;是一种具有函数优先的轻量级&#xff0c;解释型或即时编译型的编程语言。虽然它是作为开发Web页面的脚本语言而出名&#xff0c;但是它也被用到了很多非浏览器环境中&#xff0c;JavaScript基于原型编程、多范式…

Qt Creator可视化交互界面exe快速入门5

上一期介绍了加法计算器,本期介绍QObject定时器。 首先一样先建个工程,比如我这项目名为QObject 本期的任务就是制作图片在界面上显示,然后每秒定时切换,点击另一个暂停按钮,可以定格当前图片,即取消定时切换功能。 显示图片的我们可以使用显示里面的label 这个用于显示…