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虚拟机的实现、类的元信息、运行时常量池、字符串常量池、静态变量的存储等内容,希望对大家有所帮助。

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

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

相关文章

App分发测试平台:改变应用开发的游戏规则

App分发测试平台是一个提供应用开发者上传、测试、分享和发布应用的在线服务平台。它为开发者提供了一个高效的测试环境&#xff0c;并为用户提供了一个方便的应用获取渠道。其中&#xff0c;测试环节尤为关键&#xff0c;因为它能够确保应用在上线之前达到预期的功能和性能。 …

问题:Feem无法发送信息OR无法连接(手机端无法发给电脑端)

目录 前言 问题分析 资源、链接 其他问题 前言 需要在小米手机、华为平板、Dell电脑之间传输文件&#xff0c;试过安装破解的华为电脑管家、小米的MIUI文件传输等&#xff0c;均无果。&#xff08;小米“远程管理”ftp传输倒是可以&#xff0c;但速度太慢了&#xff0c;且…

Java JVM 堆、栈、方法区详解

目录 1. 栈 2. 堆 3. 方法区 4. 本地方法栈 5. 程序计数器 首先来看一下JVM运行时数据区有哪些。 1. 栈 在介绍JVM栈之前&#xff0c;先了解一下 栈帧 概念。 栈帧&#xff1a;一个栈帧随着一个方法的调用开始而创建&#xff0c;这个方法调用完成而销毁。栈帧内存放者方…

ROS学习笔记8——实现ROS通信时的常用命令

机器人系统中启动的节点少则几个&#xff0c;多则十几个、几十个&#xff0c;不同的节点名称各异&#xff0c;通信时使用话题、服务、消息、参数等等都各不相同&#xff0c;一个显而易见的问题是: 当需要自定义节点和其他某个已经存在的节点通信时&#xff0c;如何获取对方的话…

gitgud.io+Sapphire注册账号教程

gitgud.io是一个仓库&#xff0c;地址 https://gitgud.io/&#xff0c;点进去之后会看到注册页面。 意思是需要通过注册这个Sapphire账户来登录。点击右边的Sapphire&#xff0c;就跳转到Sapphire的登陆页面&#xff0c;点击创建新账号&#xff0c;就进入注册页面。&#xff0…

SpiderFlow爬虫平台漏洞利用分析(CVE-2024-0195)

1. 漏洞介绍 SpiderFlow爬虫平台项目中spider-flow-web\src\main\java\org\spiderflow\controller\FunctionController.java文件的FunctionService.saveFunction函数调用了saveFunction函数&#xff0c;该调用了自定义函数validScript&#xff0c;该函数中用户能够控制 functi…

Spring | Spring中的Bean--下

Spring中的Bean: 4.Bean的生命周期5.Bean的配装配式 ( 添加Bean到IOC容器的方式 依赖注入的方式 )5.1 基于XML的配置5.2 基于Annotation (注解) 的装配 (更常用&#xff09;5.3 自动装配 4.Bean的生命周期 Spring容器可以管理 singleton作用域的Bean的生命周期&#xff0c;在此…

go语言(七)----slice的声明方式

1、声明方式一 //声明一个slice1是一个切片&#xff0c;但是并没有给slice分配空间var slice1 []intslice1 make([]int,3)2、声明方式二 声明一个slice切片&#xff0c;同时给slice分配空间&#xff0c;3个空间&#xff0c;初始化值是0var slice1 []int make([]int,3)3、声…

ICCV2023 | PTUnifier+:通过Soft Prompts(软提示)统一医学视觉语言预训练

论文标题&#xff1a;Towards Unifying Medical Vision-and-Language Pre-training via Soft Prompts 代码&#xff1a;https://github.com/zhjohnchan/ptunifier Fusion-encoder type和Dual-encoder type。前者在多模态任务中具有优势&#xff0c;因为模态之间有充分的相互…

从临床和科研场景分析ChatGPT在医疗健康领域的应用可行性

2023年4月发表在Journal Medical Systems的文献《Evaluating the Feasibility of ChatGPT in Healthcare: An Analysis of Multiple Clinical and Research Scenarios》&#xff08;评估 ChatGPT 在医疗健康领域的可行性&#xff1a;对多种临床和研究场景的分析&#xff09;介绍…

IPv6自动隧道---6to4中继

6to4中继 普通IPv6网络需要与6to4网络通过IPv4网络互通,这可以通过6to4中继路由器方式实现。所谓6to4中继,就是通过6to4隧道转发的IPv6报文的目的地址不是6to4地址,但转发的下一跳是6to4地址,该下一跳为路由器我们称之为6to4中继。隧道的IPv4目的地址依然从下一跳的6to4地…

PPT 编辑模式滚动页面不居中

PPT 编辑模式滚动页面不居中 目标&#xff1a;编辑模式下适应窗口大小、切换页面居中显示 调整视图大小&#xff0c;编辑模式通过Ctrl 鼠标滚轮 或 在视图菜单中点击适应窗口大小。 2. 翻页异常&#xff0c;调整视图大小后&#xff0c;PPT翻页但内容不居中或滚动&#xff0c…

『MySQL快速上手』-⑩-索引特性

文章目录 1.索引的作用2.索引的理解建立测试表插入多条记录查看结果 2.1 MySQL与磁盘交互的基本单位2.1 为何IO交互要是 Page2.3 理解单个Page2.4 理解多个Page2.5 页目录2.6 单页情况2.7 多页情况2.8 B vs B2.9 聚簇索引 vs 非聚簇索引非聚簇索引聚簇索引 3.索引操作3.1 创建主…

pytest + allure(windows)安装

背景 软硬件环境&#xff1a; windows11&#xff0c;已安装anaconda&#xff0c;python&#xff0c;pycharm用途&#xff1a;使用pytest allure 生成报告allure 依赖java&#xff0c;点击查看java安装教程 allure 下载与安装 从 allure下载网址下载最新版本.zip文件 放在自…

基于YOLOv8深度学习的葡萄簇目标检测系统【python源码+Pyqt5界面+数据集+训练代码】目标检测、深度学习实战

《博主简介》 小伙伴们好&#xff0c;我是阿旭。专注于人工智能、AIGC、python、计算机视觉相关分享研究。 ✌更多学习资源&#xff0c;可关注公-仲-hao:【阿旭算法与机器学习】&#xff0c;共同学习交流~ &#x1f44d;感谢小伙伴们点赞、关注&#xff01; 《------往期经典推…

【llm 微调code-llama 训练自己的数据集 一个小案例】

这也是一个通用的方案&#xff0c;使用peft微调LLM。 准备自己的数据集 根据情况改就行了&#xff0c;jsonl格式&#xff0c;三个字段&#xff1a;context, answer, question import pandas as pd import random import jsondata pd.read_csv(dataset.csv) train_data data…

pyspark 笔记:窗口函数window

窗口函数相关的概念和基本规范可以见&#xff1a;pyspark笔记&#xff1a;over-CSDN博客 1 创建Pyspark dataFrame from pyspark.sql.window import Window import pyspark.sql.functions as F employee_salary [("Ali", "Sales", 8000),("Bob&qu…

USACO介绍 报名流程 成绩查询方式详解(文末有备赛资料)

USACO美国计算机奥林匹克活动 2023-2024新赛季的时间线安排是怎么样的&#xff1f; 2023-2024USACO竞赛时间 一般来说&#xff0c;USACO竞赛时间在12月-3月期间&#xff0c;每月都有一场比赛每次3-5小时&#xff0c;并在规定时间内完成3-4道题。23-24年USACO竞赛时间安排如下&a…

uniapp h5 生成 ubuntu桌面程序 并运行方法

uniapp h5 生成 ubuntu桌面程序 并运行方法,在window环境下开发&#xff0c;发布到ubuntu桌面&#xff0c;并运行 1、安装Nodejs 安装包官方下载地址&#xff1a;https://www.nodejs.com.cn/ 安装完后cmd&#xff0c;如图&#xff0c;即安装成功 2、通过Nodejs安装 electron…

[flutter]GIF速度极快问题的两种解决方法

原因&#xff1a; 当GIF图没有设置播放间隔时间时&#xff0c;电脑上会默认间隔0.1s&#xff0c;而flutter默认0s。 解决方法一&#xff1a; 将图片改为webp格式。 解决方法二&#xff1a; 为图片设置帧频率&#xff0c;添加播放间隔。例如可以使用GIF依赖组件设置每秒运行…