JVM工作原理与实战(十九):运行时数据区-方法区
专栏导航
JVM工作原理与实战
RabbitMQ入门指南
从零开始了解大数据
目录
专栏导航
前言
一、运行时数据区
二、方法区
1.方法区介绍
2.方法区在Java虚拟机的实现
3.类的元信息
4.运行时常量池
5.字符串常量池
6.静态变量的存储
总结
前言
JVM作为Java程序的运行环境,其负责解释和执行字节码,管理内存,确保安全,支持多线程和提供性能监控工具,以及确保程序的跨平台运行。本文主要介绍了方法区、方法区在Java虚拟机的实现、类的元信息、运行时常量池、字符串常量池、静态变量的存储等内容。
一、运行时数据区
Java虚拟机(JVM)在运行Java程序期间,会创建并维护一系列内存区域,这些区域总称为运行时数据区。这些区域根据其用途和特性,被严格定义并管理。《Java虚拟机规范》详细规定了这些区域的作用和行为,以确保所有Java虚拟机实现的一致性和正确性。
线程不共享区域:
- 程序计数器:用于存储当前线程执行的字节码指令地址。这个区域是每个线程独有的,不共享。
- Java虚拟机栈:每个线程在创建时都会创建一个虚拟机栈,每个方法调用都会创建一个栈帧,用于存储局部变量、操作数栈、动态链接和方法出口信息。
- 本地方法栈:与虚拟机栈相似,本地方法栈为native方法提供服务。
线程共享区域:
- 方法区:用于存储已被JVM加载的类信息、常量、静态变量以及即时编译器编译后的代码等数据。
- 堆:堆是所有线程共享的区域,用于动态分配内存。所有的对象实例以及数组都应当在堆上分配。

二、方法区
1.方法区介绍
方法区是Java虚拟机中的一部分,用于存储已被虚拟机加载的类信息、常量、静态变量以及即时编译器编译后的代码等数据。这个区域的设计目标是为所有线程提供共享的、动态类型的数据。它的核心功能是支持类的加载和链接,以及提供运行时类型信息。
方法区主要由以下三部分构成:
- 类的元信息(Class Metadata):这部分保存了关于类的所有基本信息。这些信息在类加载时被创建,并存储在方法区中。类的元信息包括类的名称、类的访问权限、类的字节码版本、父类的名称、实现的接口列表、字段和方法信息等。这些信息在运行时被JVM使用,以支持类的方法解析、反射操作和动态代理等功能。
- 运行时常量池(Runtime Constant Pool):运行时常量池是方法区中的一个重要部分,它主要负责存储字节码文件中的常量池内容。常量池是字节码文件中的一个特殊部分,用于存储各种类型的常量,如字面量、类符号引用等。在运行时,JVM会根据需要动态地向常量池中添加、删除或修改相应的常量。此外,对于每个类,其常量池都有一个私有的副本,但所有线程共享同一个运行时常量池,以确保线程安全。
- 字符串常量池(String Constant Pool):字符串常量池是运行时常量池的一部分,用于存储字符串字面量。在Java程序中,每个独特的字符串字面量都会在字符串常量池中创建一个相应的字符串对象。如果一个字符串字面量已经存在于常量池中,则不会创建新的对象,而是返回对已有对象的引用。这种设计可以有效地节省内存,并避免创建重复的字符串对象。
2.方法区在Java虚拟机的实现
方法区是Java虚拟机结构中的重要部分,它是《Java虚拟机规范》中定义的一个抽象概念。每款具体的Java虚拟机实现可能会根据规范进行不同的优化和调整。以HotSpot虚拟机为例,来探讨其实现细节。
在JDK7及更早的版本中,方法区被实现为永久代(PermGen)。它位于Java堆内存区域中,并通过虚拟机参数-XX:MaxPermSize来控制其最大大小。然而,这种实现方式在JDK8中被彻底改变。
从JDK8开始,方法区被移至元空间(Metaspace)中。元空间不再属于Java堆的一部分,而是直接建立在操作系统的本地内存中。这种设计使得元空间的大小不再受Java堆大小的限制,而是取决于本地内存的大小。可以通过虚拟机参数-XX:MaxMetaspaceSize可以设置元空间的最大大小。

在诊断和监控Java应用时,了解这些差异尤为重要。使用如Arthas这样的诊断工具,可以查看不同版本的Java虚拟机的内存使用情况。在JDK7中,需要关注ps_perm_gen属性来查看方法区的使用情况;而在JDK8及之后的版本,则需要查看metaspace属性。
案例:
为了模拟方法区的溢出情况,可以使用如ByteBuddy这样的框架动态生成大量的字节码数据,并观察方法区是否会出现内存溢出。
引入ByteBuddy依赖:
<dependency><groupId>net.bytebuddy</groupId><artifactId>byte-buddy</artifactId><version>1.12.23</version></dependency>编写测试案例:
public class Demo1 extends ClassLoader {public static void main(String[] args) throws IOException {System.in.read();Demo1 demo1 = new Demo1();int count = 0;while (true) {String name = "Class" + count;ClassWriter classWriter = new ClassWriter(0);classWriter.visit(Opcodes.V1_8, Opcodes.ACC_PUBLIC, name, null, "java/lang/Object", null);byte[] bytes = classWriter.toByteArray();demo1.defineClass(name, bytes, 0, bytes.length);System.out.println(++count);}} }在JDK7上,执行十几万次操作就可能出现错误;而在JDK8上,尽管内存使用量会直线上升,但程序并不会出现错误。
3.类的元信息
方法区,也被称为元空间,是Java虚拟机中用于存储类的元数据信息的区域。这些元信息,通常被称为InstanceKlass对象,包含了关于类的基本信息,例如类的名称、父类的名称、实现的接口、成员变量和方法等。这些信息在类的加载阶段被完全建立并存储在方法区中。
InstanceKlass对象是类元信息的核心,它不仅包含了类的静态信息,如字段和方法,还包含了类的动态行为信息,如字节码信息和常量池等。这些信息在运行时被JVM用于支持诸如反射、动态类加载和异常处理等功能。

需要注意的是,方法区的实现和组织方式可能会因JVM的实现而有所不同。例如,在一些JVM实现中,方法区可能会被组织成一个或多个哈希表,以快速查找类的元信息。而在其他实现中,可能会使用其他数据结构或算法来组织和管理这些信息。
4.运行时常量池
在Java的内存区域中,方法区用于存储类的元数据信息。然而,除了存储类的元信息之外,方法区还包含了一个重要的部分:运行时常量池。
运行时常量池是Java虚拟机在运行时创建的一个数据结构,用于存储字节码中的常量池内容。常量池是字节码文件中的一个特殊部分,包含了程序中的常量值,例如字符串字面量、整数等。这些常量在字节码文件中通过编号索引的方式进行访问,这种常量池称为静态常量池。
当字节码文件被加载到内存中时,静态常量池的内容会被复制到运行时常量池中。与静态常量池不同,运行时常量池中的常量可以直接通过内存地址进行访问,因此具有更高的访问速度。这种运行时常量池的设计使得Java程序在运行时能够快速地访问和操作常量,提高了程序的执行效率。

5.字符串常量池
在运行时数据区的方法区中,除了类的元信息和运行时常量池外,还有一个特别重要的区域,那就是字符串常量池。字符串常量池主要负责存储字节码文件中定义的常量字符串内容。例如,在代码中定义的常量字符串“123”,这个字符串就会存放在字符串常量池中。

早期的设计中,字符串常量池被视为运行时常量池的一部分,它们共享相同的存储空间。然而,随着技术的进步和优化需求的变化,Java虚拟机对这两者的存储区域进行分离。这种调整的主要原因在于,字符串常量池和运行时常量池的功能和职责存在明显的差异。运行时常量池主要负责管理动态编译和类的加载,而字符串常量池则专注于存储和管理程序中定义的常量字符串。

6.静态变量的存储
在Java的运行时数据区中,静态变量(也称为类变量)的存储位置在不同版本的Java中有所不同。
在JDK 6及之前的版本中,静态变量是存放在方法区中的,确切地说,是在永久代(PermGen)中。然而,随着Java的发展,这个设计逐渐暴露出一些问题,例如内存溢出和空间限制等。
从JDK 7开始,Java对运行时数据区进行了重新设计,其中最显著的变化就是将静态变量从永久代移出,并存放在了堆内存中的Class对象中。这种变化不仅解决了永久代存在的问题,还使得静态变量的存储更加符合其作为类级别的变量的特性。每个Class对象都包含了该类的静态变量信息,这些信息随着类的加载而加载到内存中。这种设计使得静态变量的访问更加直接和高效,因为它们不再需要从方法区中查找,而是可以直接从Class对象中获取。

总结
JVM是Java程序的运行环境,负责字节码解释、内存管理、安全保障、多线程支持、性能监控和跨平台运行。本文主要介绍了方法区、方法区在Java虚拟机的实现、类的元信息、运行时常量池、字符串常量池、静态变量的存储等内容,希望对大家有所帮助。
相关文章:
JVM工作原理与实战(十九):运行时数据区-方法区
专栏导航 JVM工作原理与实战 RabbitMQ入门指南 从零开始了解大数据 目录 专栏导航 前言 一、运行时数据区 二、方法区 1.方法区介绍 2.方法区在Java虚拟机的实现 3.类的元信息 4.运行时常量池 5.字符串常量池 6.静态变量的存储 总结 前言 JVM作为Java程序的运行环境…...
webassembly003 whisper.cpp的项目结构CMakeLists.txt
注:带星号的为非重要部分 基础配置 cmake_minimum_required (VERSION 3.5)project(whisper.cpp VERSION 1.5.0)# Add path to modules list(APPEND CMAKE_MODULE_PATH "${CMAKE_CURRENT_SOURCE_DIR}/cmake/") # 在\cmake文件夹下还有BuildTypes.cmake&a…...
克魔助手工具详解、数据包抓取分析、使用教程
目录 摘要 引言 克魔助手界面 克魔助手查看数据捕获列表 数据包解析窗口 数据包数据窗口 克魔助手过滤器表达式的规则 抓包过滤器实例 总结 参考资料 摘要 本文介绍了克魔助手工具的界面和功能,包括数据包的捕获和分析,以及抓包过滤器的使用方…...
【Docker】contos7安装 Nacos容器部署单个部署集群
🎉🎉欢迎来到我的CSDN主页!🎉🎉 🏅我是平顶山大师,一个在CSDN分享笔记的博主。📚📚 🌟推荐给大家我的博客专栏《Docker】contos7安装 Nacos容器部署单个&…...
UML-通信图和交互概览图(通信图和顺序图的区别与联系)
UML-通信图和交互概览图(通信图和顺序图的区别与联系) 一、通信图简介1.消息2.链接 二、通信图和[顺序图](https://blog.csdn.net/weixin_65032328/article/details/135587782)的联系与区别三、交互概览图四、顺序图转化为通信图练习 一、通信图简介 通…...
Linux 使用PS命令掌握进程管理
在Linux系统中,进程管理是系统管理员和开发人员必备的技能之一。而PS命令作为进程管理的重要工具,可以帮助我们查看和监控系统中运行的进程。本文将详细解析PS命令的使用方法和输出结果,帮助读者全面掌握进程管理的利器。 PS命令概述…...
Debian 10.13.0 安装图解
引导和开始安装 这里直接回车确认即可,选择图形化安装方式。 选择语言 这里要区分一下,当前选中的语言作为安装过程中安装器所使用的语言,这里我们选择中文简体。不过细心的同学可能发现,当你选择安装器语言之后,后续安…...
SQLite 3.45.0 发布!
SQLite 开发团队于 2024 年 01 月 18 日发布了 SQLite 3.45.0 版本,带来了一些 JSON 和优化器增强,让我们一睹为快! JSON 函数 SQLite 3.45.0 版本开始,所有的 JSON 函数将会使用全新的内部格式存储 JSON 数据,也就是…...
MongoDB聚合:$set
聚合$set阶段可以为文档添加新的字段。$set输出的文档包含输入文档中的所有现有字段和新添加的字段。$set是$addFields的别名,从MongoDB4.2开始支持。$set和$addFields等价于$project阶段,这两个阶段都等同于 $project 阶段,后者明确指定输入…...
《Python数据分析技术栈》第01章 02 Jupyter入门(Getting started with Jupyter notebooks)
02 Jupyter入门(Getting started with Jupyter notebooks) 《Python数据分析技术栈》第01章 02 Jupyter入门(Getting started with Jupyter notebooks) Before we discuss the essentials of Jupyter notebooks, let us discuss…...
【征服redis5】redis的Redisson客户端
目录 1 Redisson介绍 2. 与其他Java Redis客户端的比较 3.基本的配置与连接池 3.1 依赖和SDK 3.2 配置内容解析 4 实战案例:优雅的让Hash的某个Field过期 5 Redisson的强大功能 1 Redisson介绍 Redisson 最初由 GitHub 用户 “mrniko” 创建,并在…...
React16源码: React中的beginWork的源码实现
beginWork 1 )概述 在 renderRoot 之后,要对我们的 Fiber 树每一个节点进行对应的更新更新节点的一个入口方法,就是 beginWork这个入口方法会有帮助我们去优化整棵树的更新过程 react 它的节点其实是非常多的,如果每一次子节点的…...
5-微信小程序语法参考
1. 数据绑定 官网传送门 WXML 中的动态数据均来自对应 Page 的 data。 数据绑定使用 Mustache 语法(双大括号)将变量包起来 ts Page({data: {info: hello wechart!,msgList: [{ msg: hello }, { msg: wechart }]}, })WXML <view class"vie…...
数组练习 Leetcode 566.重塑矩阵
在 MATLAB 中,有一个非常有用的函数 reshape ,它可以将一个 m x n 矩阵重塑为另一个大小不同(r x c)的新矩阵,但保留其原始数据。 给你一个由二维数组 mat 表示的 m x n 矩阵,以及两个正整数 r 和 c &#…...
Linux centos中find命令的多种用途:按照具体应用来详细说明find的用法举例
目录 一、find命令 二、find命令的语法 (一)语法格式 (二)选项 1、选项(option)介绍 2、控制符号链接的option 3、调试选项debugopts 4、优化选项 (三)表达式expression 1、选项options 2、测试…...
服务器数据恢复—OceanStor存储raid5热备盘同步数据失败的数据恢复案例
服务器数据恢复环境: 华为OceanStor某型号存储,存储内有一组由24块硬盘组建的raid5阵列,配置1块热备盘。 服务器故障: 该存储raid5阵列中有一块硬盘离线,热备盘自动激活并开始同步数据,在热备盘同步数据的…...
Xline v0.6.1: 一个用于元数据管理的分布式KV存储
Xline是什么?我们为什么要做Xline? Xline是一个基于Curp协议的,用于管理元数据的分布式KV存储。现有的分布式KV存储大多采用Raft共识协议,需要两次RTT才能完成一次请求。当部署在单个数据中心时,节点之间的延迟较低&a…...
【CSS】解决height = line-height 文字不垂直居中(偏上、偏下)的问题
解决办法1: 查看 font-family 属性,确认是否是因为字体而导致的不垂直居中问题。 其他小知识: 基线就是小写x字母的下边缘(线) 就是我们常说的 基线。line-height 属性设置的行高也就是定义的两行文字基线之间的距离! 参考文章:…...
天津想转行学python培训班靠谱吗?
现在的职业如此繁多,很多人把高薪当成衡量工作好坏的重要标准,因此IT行业以超出其他行业几倍薪资水平成为不错的选择,而Python又以其简单易学好上手成为大家所青睐的学习目标。 Python发展前景如何 Python语言就业发展方向广泛:…...
(C语言)冒泡排序
一、运行结果; 二、源代码; # define _CRT_SECURE_NO_WARNINGS # include <stdio.h>//实现buble_sort函数; void buble_sort(int arr[], int sz) {//初始化变量值;int i 0;//嵌套循环冒泡排序;//外层循环&…...
SPEC CPU 2017基准测试深度解析:从原理到实战调优
1. 项目概述:一次性能基准测试的巅峰对决最近在服务器和芯片圈子里,一个消息炸开了锅:曙光服务器在SPEC CPU 2017基准测试中,一口气刷新了四项世界纪录。对于圈外人来说,这可能只是一条普通的科技新闻,但对…...
cursor接入外部大模型教程!新手必看
一、接入前准备 在开始之前,请先登录你的大模型平台,这里使用 我自用的举例官网地址,创建并复制你的 API Key。 这里添加令牌,有名称和分组,简单举例,填入名称 cursor-claude, 一个key只能选一…...
为什么所有人都在聊RAG?看这篇,小白也能彻底搞懂
你是否有过这样的经历——你满怀期待地问 AI 一个专业问题,它流畅地给了你一段"答案",引经据典、逻辑自洽。 结果一查,发现全是错的。一本正经地胡说八道。 这就是大语言模型(LLM)的致命短板:它…...
OBS背景移除插件:从零到一的AI虚拟背景终极指南 [特殊字符]
OBS背景移除插件:从零到一的AI虚拟背景终极指南 🎬 【免费下载链接】obs-backgroundremoval An OBS plugin for removing background in portrait images (video), making it easy to replace the background when recording or streaming. 项目地址: …...
影刀RPA跨境店群运营架构:Python协同Chromium底层调度与高并发容器化架构
定了。在这场旷日持久的跨境电商反爬风控拉锯战中,我们终于用一套基于 Python 深度协同的分布式微服务调度架构,重塑了跨境千店矩阵的自动化底座。 这几天,科技圈被“DeepSeek V4 首发华为昇腾芯片,国产 AI 开始打破英伟达 CUDA …...
告别刷机兼容性噩梦:AnyKernel3如何让Android内核适配变得轻松
告别刷机兼容性噩梦:AnyKernel3如何让Android内核适配变得轻松 【免费下载链接】AnyKernel3 AnyKernel, Evolved 项目地址: https://gitcode.com/gh_mirrors/an/AnyKernel3 还在为不同Android设备的内核适配而烦恼吗?每次发布新内核都要为不同ROM…...
高性能自动化网页信息提取工具实战指南:大规模目标扫描与安全检测技术方案
高性能自动化网页信息提取工具实战指南:大规模目标扫描与安全检测技术方案 【免费下载链接】URLFinder 一款快速、全面、易用的页面信息提取工具,可快速发现和提取页面中的JS、URL和敏感信息。 项目地址: https://gitcode.com/gh_mirrors/ur/URLFinder…...
利用 Taotoken 多模型选型能力优化智能客服对话场景
🚀 告别海外账号与网络限制!稳定直连全球优质大模型,限时半价接入中。 👉 点击领取海量免费额度 利用 Taotoken 多模型选型能力优化智能客服对话场景 对于智能客服系统的开发者而言,选择合适的对话模型是平衡服务效果…...
嵌入式存储优化实战:如何为你的AUTOSAR FEE模块选择合适的FeeMainFunctionPeriod与FeeMaxBytesPerCycle?
嵌入式存储优化实战:AUTOSAR FEE模块参数配置与性能调优 在汽车电子控制单元(ECU)的开发中,存储管理一直是影响系统性能和可靠性的关键因素。AUTOSAR的Flash EEPROM Emulation(FEE)模块作为非易失性数据存储…...
终极HTML转Figma教程:3步将任何网站变为可编辑设计稿
终极HTML转Figma教程:3步将任何网站变为可编辑设计稿 【免费下载链接】figma-html Convert any website to editable Figma designs 项目地址: https://gitcode.com/gh_mirrors/fi/figma-html 你是否曾为网站设计稿与最终实现不一致而烦恼?或者作…...
