当前位置: 首页 > news >正文

JVM之堆和方法区

目录

1.堆

1.1 堆的结构

1.1.1 新生代(Young Generation)

1.1.2 年老代(Old Generation)

1.1.3 永久代/元空间(Permanent Generation/Metaspace)

 1.2 堆的内存溢出

1.3 堆内存诊断

1.3.1 jmap

1.3.2 jvisualvm

2. 方法区

2.1 方法区的结构


1.堆

JVM堆是Java程序运行时内存管理的核心,它主要用于存储对象实例和数组。堆内存的特点是动态分配和回收,它允许对象的创建和销毁,同时也需要注意内存泄漏和性能问题。

1.1 堆的结构

JVM堆通常被分为三个主要部分:

1.1.1 新生代(Young Generation)

新生代用于存储刚刚被创建的对象。它被分为三个区域:Eden空间和两个Survivor空间(通常称为From和To空间)。新创建的对象首先被分配到Eden空间,经过一次垃圾回收后,仍然存活的对象会被移到Survivor空间。多次循环后,仍然存活的对象会被移到年老代。

1.1.2 年老代(Old Generation)

年老代用于存储生命周期较长的对象。在新生代中经过多次垃圾回收后,仍然存活的对象会被晋升到年老代。年老代中的对象一般需要经历更多的垃圾回收周期才会被回收。

1.1.3 永久代/元空间(Permanent Generation/Metaspace)

在早期的JVM版本中,永久代用于存储类的元数据、方法信息以及静态变量。然而,由于永久代容易导致内存泄漏和溢出问题,1.8后JVM引入了元空间来代替。元空间的元数据存储在本地内存中,不再受到固定大小的限制。

jdk1.8以及之后:在堆内存中,逻辑上存在,物理上不存在(元空间使用的是本地内存),如下图:

什么是永久代和元空间?

方法区是一种规范,不同的虚拟机厂商可以基于规范做出不同的实现,永久代和元空间就是出于不同jdk版本的实现。
方法区就像是一个接口,永久代与元空间分别是两个不同的实现类。
只不过永久代是这个接口最初的实现类,后来这个接口一直进行变更,直到最后彻底废弃这个实现类,由新实现类—元空间进行替代。

 1.2 堆的内存溢出

使用如下代码:

public class a {public static void main(String[] args)  {List<String> list=new ArrayList<>();String a="hello";while (true){a=a+a;list.add(a);}}
}

执行以上代码后就会发生堆内存溢出如下图:

Java堆的大小可以通过命令行参数来配置,主要参数包括:

  • -Xms:设置堆的初始大小。
  • -Xmx:设置堆的最大大小。

通常,将这两个参数设置为相同的值可以减少堆的动态调整,提高性能。例如:

java -Xms512m -Xmx512m -jar YourApp.jar

这将设置堆的初始大小和最大大小都为512兆字节。

1.3 堆内存诊断

1.3.1 jmap

首先jps找到java运行的进程,然后jmap -heap 进程id就可以查看堆内存了,如下图:

1.3.2 jvisualvm

执行下面的代码:

public class a {public static void main(String[] args) throws InterruptedException {Thread.sleep(30000);byte[] bytes = new byte[1024 * 1024 * 50];System.out.println("-------");Thread.sleep(30000);bytes=null;System.out.println("-------");Thread.sleep(30000);}
}

然后执行jvisualvm会得出如下图:当bytes=null后进行垃圾回收后,内存占用直接减少50M。如下图:

2. 方法区

方法区(Method Area)是JVM的另一个重要内存区域,它主要用于存储类的元数据、静态变量、常量池以及方法代码。

2.1 方法区的结构

方法区包含以下主要部分:

类的元数据

方法区存储了每个类的元数据,包括类名、父类、接口、字段、方法等信息。这些信息在程序运行时起到重要作用,例如方法的调用和字段的访问。

 静态变量

静态变量属于类,不属于对象的实例。这些变量在类加载时初始化,存在于方法区中。

常量池

常量池包含了类中使用的常量,例如字符串、数字、类名等。它为类的运行时常量提供了存储空间。

方法代码

方法区存储了类中的方法代码,包括字节码指令和方法的字节码表示。这些代码在方法被调用时执行。

在早期的JVM版本中,方法区被实现为永久代。然而,由于永久代的内存泄漏和性能问题,JVM在较新的版本中引入了元空间来替代永久代。元空间的好处是不再受限于固定大小,避免了永久代引起的一些问题。如下图:

http://www.lryc.cn/news/155660.html

相关文章:

  • Java 中的 IO 和 NIO
  • Linux-crontab使用问题解决
  • 【设计模式】
  • 2023_Spark_实验四:SCALA基础
  • 【深入解析spring cloud gateway】04 Global Filters
  • c++搜索基础进阶
  • 管网水位监测的必要性
  • 无涯教程-Android - 系统架构
  • await接受成功的promise,失败的promise用try catch
  • 赞奇科技参与华为云828 B2B企业节,云工作站入选精选产品解决方案
  • Docker私有镜像仓库(Harbor)安装
  • 【深入解析spring cloud gateway】06 gateway源码简要分析
  • 2023年行研行业研究报告
  • linux上vscode中.cpp文件中引入头文件.hpp时报错:找不到头文件(启用错误钵形曲线)
  • Sphinx Docstring
  • JVM的故事——虚拟机类加载机制
  • Sentry 是一个开源的错误监控和日志聚合平台-- 通过docker-compose 安装Sentry
  • Redis 7 第六讲 主从模式(replica)架构篇
  • 学习资源记录 =0=
  • Python import包路径管理
  • OB Cloud助力泡泡玛特打造新一代分布式抽盒机系统
  • Linux socket网络编程实战(tcp)实现双方聊天
  • BuhoCleaner for mac:让你的Mac重获新生
  • 陶氏公司将出席2023第二届中国汽车碳中和峰会
  • 【linux命令讲解大全】051.Linux Awk脚本语言中的字段定界符和流程控制
  • Gradle下载安装教程
  • Python 之 match 表达式
  • .NET Framework 微软官方下载地址
  • OpenCV(十四):ROI区域截取
  • Java学习笔记之----I/O(输入/输出)一