JVM运行时数据区域

news2025/2/26 21:19:19

文章目录

  • 内存结构
      • 程序计数器(寄存器)
      • 虚拟机栈
        • 局部变量表
          • 两类异常状况
        • 线程运行诊断
      • 本地方法栈
      • 方法区
      • 运行时常量池
        • 串池(StringTable)
          • 字符串的拼接
          • 串池的位置
          • StringTable垃圾回收
          • StringTable性能调优
      • 直接内存

内存结构

程序计数器(寄存器)

Java源代码不能被cpu直接执行,需要经过编译,编译成二进制的字节码,二进制字节码中的一行行代码就是jvm指令

Java跨平台技术就是靠这一条条jvm指令,对任何操作系统都是一致的

这些指令再经过解释器解释成机器码,机器码可以被cpu执行

程序计数器的作用就是在解释器解释jvm指令的过程中记住下一条jvm指令的执行地址
在物理上程序计数器是通过寄存器(cpu中读取最快的一个单元)实现的(因为读取地址是非常频繁的)

极小的一块内存

每条线程都需要一个独立的程序计数器(程序计数器是线程私有的)

是在jvm规范中,唯一一个不会存在内存溢出的区

当前线程所执行的字节码的行号指示器

字节码解释器通过改变计数器的值选取下一条要执行的字节码的指令

是程序控制流的指示器

虚拟机栈

  • 每个线程运行所需要的内存,称为虚拟机栈
  • 每个栈由栈帧构成,对应每次方法调用时所占的内存
  • 每个线程只能有一个活动栈帧,对应当前正在执行的那个方法

也是线程私有的,生命周期与线程相同,线程结束,栈结束,所以不存在垃圾回收问题

栈内存不是越大越好,栈的内存大了,线程数就少了,内存大了只能增快方法的递归调用

方法内局部变量是否具有线程安全问题:

如果这个变量是共享的,如static,就会有线程安全问题, 如果这个变量逃离了方法的作用范围,也会有线程安全问题

否则,不会有线程安全问题

每个方法被执行时,jvm会同步创建一个栈帧,用于存储局部变量表、操作数栈、动态连接、方法出口等信 息

**栈帧:**每个方法运行时需要的内存(如参数,局部变量,返回地址,这些都要提前分配内存)

8大基本类型,对象引用,实例方法

局部变量表
  • 存放了各种jvm基本数据类型,对象引用(reference类型),returnAddress类型,这些数据类型在局部变量表中的存储空间以局部变量槽(Slot),来表示。
  • 64位的long,double类型占用两个变量槽,其余数据类型占用一个。
  • 所需要的内存空间在编译期间完成分配,方法运行期间不会改变局部变量表的大小(指变量槽的数量)
两类异常状况
  • 如果线程请求的栈深度超过了虚拟机允许的深度(栈帧过多,栈帧过大),抛出StackOverflowError异常
  • 如果虚拟机栈容量可以动态扩展,当栈扩展时无法申请足够的内存抛出OutOfMemeoryError异常

HotSpot虚拟机的栈容量是不可以动态扩展的,所以在HotSpot上不会由于·虚拟机栈无法扩展而导致OutOfMemeoryError异常,但是如果线程申请栈空间失败,仍会出现OOM异常

线程运行诊断
  • cpu占用过多
  • 程序运行很长时间没有结果

本地方法栈

与虚拟机栈发挥作用非常相似,和虚拟机栈一样,也会在栈深度溢出和栈扩展失败时抛出异常StackOverflow,OOM

区别:

虚拟机栈为虚拟机执行Java方法服务,本地方法栈为虚拟机使用到的本地(Native)方法服务

VM options 控制堆内存大小为8mb -Xmx 8m

  • 虚拟机管理内存中最大的一块
  • 是被所有线程共享(堆中对象需要考虑线程安全问题)的一块内存区域,一个jvm只有一个堆内存,堆内存大小可以调节
  • 在虚拟机启动时创建
  • 唯一目的是存放对象实例,几乎所有对象实例都在这里分配内存
  • 垃圾收集器管理的内存区域,也被称为GC堆

从回收内存角度来看

现代垃圾收集器大部分都是基于分代收集理论设计的,如新生代,永久代等,而这些仅仅是一部分垃圾收集器的设计风格而已,不是虚拟机固有布局,也不是对堆的进一步细致划分

从分配内存角度看

所有线程共享的Java堆中可以划分出多个线程私有的分配缓冲区,用来提升对象分配时的效率

无论从哪个角度,堆中存储的都只能是对象的实例,将Java堆细分只是为了更好的回收内存,更快的分配内存

Java堆不需要连续的内存

Java堆既可以是固定大小的,也可以是可扩展的,目前主流的虚拟机都是按照可扩展来实现的,如果堆中没有内存完成实例分配,并且堆无法扩展时,将抛出OOM异常

方法区

  • 被所有线程共享

  • 静态变量,常量,类信息(构造方法,接口定义),运行时常量池存在方法区中,但是实例变量存在堆内存中和方法区无关(重点)

    static , final,Class模板,常量池

  • 方法区是堆的一个逻辑部分(关于他到底是不是堆的一部分,不同的jvm厂商实现方式不同),但是有个别名叫非堆,从而与Java堆区分开来

JDK8之前是使用永久代实现的方法区,考虑到HotSpot的发展,这种实现方式被逐步放弃,改为用本地内存来实现方法区

JDK7时,将原本放在永久代的字符串常量池,静态变量等移出

JDK8,完全废弃的永久代的概念,改用本地内存中实现的元空间,把JDK7中剩余的内容全部移到元空间中

和Java堆一样不需要连续的内存,可以选择固定大小和可扩展,甚至可以选择不实现垃圾收集

相对来说,方法区垃圾收集比较少见,这部分内存回收的目标是针对常量池的回收和对类型的卸载

JDK8前会导致永久代内存溢出

JDK8后会导致元空间内存溢出

运行时常量池

常量池:就是一张表,虚拟机指令根据这张常量表找到要执行的类名,方法名,参数类型,字面量等信息

  • 是方法区的一部分
  • 常量池是.class文件中的,当该类被加载,他的常量池信息就会放入运行时常量池,并把里面的符号地址变成真实的地址
串池(StringTable)

StringTable特性:

  • 常量池中的字符串仅仅是符号,只有用到时才变成对象

  • 利用串池的机制,来避免重复创建字符串对象

  • 字符串变量拼接原理是StringBuilder (1.8)

  • 字符串常量拼接原理是编译器优化

  • 可以用intern方法,主动将串池中还没有的字符串对象放入串池

    • 1.8将这个字符串对象放入串池时,如果有则不会放入,没有就将这个对象放入,最后返回串池中的对象
    • 1.6如果有不会放入,如果没有把对象复制一份放入串池,最后返回串池中的对象

    区别:1.8放的是堆中的地址,1.6放的是对象副本的地址

String对象的加载是延迟的,只有走到才会将对象放到串池

/**
 * @author gwj
 */
public class HelloWorld {
    public static void main(String[] args) {
        //常量池中的信息,运行时被加载到运行时常量池,这是a b ab都是常量池中的符号,还没有变成Java字符串对象
        String s1 = "a";
        String s2 = "b";
        String s3 = "ab";
    }
}

在这里插入图片描述

StringTable 是哈希表结构,不能扩容

字符串的拼接
public class HelloWorld {
    public static void main(String[] args) {
        //常量池中的信息,运行时被加载到运行时常量池,这是a b ab都是常量池中的符号,还没有变成Java字符串对象
        String s1 = "a";
        String s2 = "b";
        String s3 = "ab";
        String s4 = s1 + s2; //new StringBuilder().append("a").append("b").toString()
        					 //new String("ab")
        System.out.println(s3 == s4);
        //s3在串池中,s4在堆中,所以false
        //javac 在编译期间的优化,结果在编译期已确定为ab
        //而上一行代码中s1和s4是变量,只能在运行期间用StringBuilder动态拼接
        String s5 = "a" + "b";
    }
}

在这里插入图片描述

StringBuilder的toString源码,new了一个String对象

@Override
public String toString() {
    // Create a copy, don't share the array
    return new String(value, 0, count);
}

在这里插入图片描述

从字节码中能看出,在执行String s4 = “a” + “b”时,生成字节码与String s3 = “ab”相同

串池的位置
  • JDK1.6时,StringTable在常量池中,常量池在方法区中,方法区使用永久代实现
  • JDK1.8时,StringTable从永久代转到了堆中(因为串池中存着大量字符串,放在永久代中使用效率低,且永久代垃圾回收较难触发
StringTable垃圾回收

当内存空间不足时,StringTable中那些没有被引用的字符串就会被回收

StringTable性能调优
  • 如果系统中字符串常量个数非常多,建议将StringTableSize调大些,减少哈希冲突,提高查找效率
  • 可以通过intern方法让字符串入池,从而减少重复字符串的个数,减少内存的占用

直接内存

  • 直接内存不是虚拟机运行时数据区的一部分,也不是定义的内存区域,是操作系统内存,但是也被频繁使用,也可能导致OOM异常

  • 不受jvm内存回收管理,直接内存的释放可以通过调用Unsafe对象的freeMemory方法

  • 常见于NIO操作中,用于数据缓冲区

  • JDK4中加入了NIO类,引入了一种基于通道和缓冲区的I/O方式

  • 可以使用Native函数直接分配堆外内存,然后通过一个存储在Java堆中的DirectByteBuffer对象作为这块内存的引用进行操作,这样避免了在Java堆和Native堆中来回复制数据

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

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

相关文章

关于标准库中的vector - (涉及迭代器失效,深浅拷贝,构造函数,内置类型构造函数,匿名对象)

目录 关于vector vector中的常见接口 vector常见接口的实现 迭代器失效 关于深浅拷贝 关于vector 关于vector的文档介绍 1. vector是表示可变大小数组的序列容器。 2. 就像数组一样,vector也采用的连续存储空间来存储元素。也就是意味着可以采用下标对vector的元…

【ArcGIS Pro微课1000例】0040:ArcGIS Pro创建北极点、南极点

文章目录 一、创建北极点图层二、创建北极点三、不同投影系下北极点的位置一、创建北极点图层 选择一个数据库,在上面右键→新建→要素类。 输入名称:北极点。 空间参考:WGS 1984 点击创建。 二、创建北极点 在编辑选项卡下,点击【创建】。 在创建要素窗口中,点击北极点…

docker容器中创建非root用户

简介 用 docker 也有一段时间了,一直在 docker 容器中使用 root 用户肆意操作。直到部署 stable diffusion webui 我才发现无法使用 root 用户运行它,于是才幡然醒悟:是时候搞个非 root 用户了。 我使用的 docker 镜像文件是 centos:centos…

FL Studio Producer Edition21.0.3中文版安装详解(附下载链接)

fl studio 21中文版是Image-Line公司继20版本之后更新的水果音乐制作软件,很多用户不太理解,为什么新版本不叫fl studio 21或fl studio2024,非得直接跳到21.2版本,其实该版本是为了纪念该公司22周年,所以该版本也是推出…

(C++)盛水最多的容器--双指针法

个人主页:Lei宝啊 愿所有美好如期而遇 力扣(LeetCode)官网 - 全球极客挚爱的技术成长平台备战技术面试?力扣提供海量技术面试资源,帮助你高效提升编程技能,轻松拿下世界 IT 名企 Dream Offer。https://le…

pgsql分别获取日期中的年、月、日,并处理前台展示有小数点的情况

使用extract()函数 select extract(YEAR from 需要处理的日期字段) from tablename; --获取年份 select extract(MONTH from 需要处理的日期字段) from tablename; --获取月份 select extract(DAY from 需要处理的日期字段) from tablename; --获取日 实际应用:…

05:2440----代码重定义

目录 一:引入 1:基本概念 2:NAND启动 3:NOR启动 4:变量 5:实验证明 A:代码makefile B:NOR启动 C:NAND启动 D:内存空间 二:链接脚本 1:NOR 2:NAND 3:解决方法 A:尝试解决 B:方法一解决 A:简…

clickhouse的向量化执行

背景 clickhouse快的很大一部分原因来源于数据的向量化执行,本文就来看一下向量化执行和正常标量执行的区别 SIMD的向量化执行 从上图可知,clickhouse通过SIMD指令可以做到一个cpu周期操作两个向量的运算操作,比起普通的cpu指令效率提高了N…

一些ab命令

1.ab简介 ab是apache自带的压力测试工具,是apachebench命令的缩写。ab非常实用,它不仅可以对apache服务器进行网站访问压力测试,也可以对或其它类型的服务器如nginx、tomcat、IIS等进行压力测试。 ab的原理:ab命令会创建多个并发…

Redis分布式锁学习总结

⭐️ 前言 想必大家都有过并发编程的经验,在一个单体应用中,可以通过java提供的各种锁机制来控制多线程对于单体应用中同一资源的并发访问;那么在分布式场景下,想要控制多个应用对于同一外部资源的并发访问,就要用到分…

npm ERR! notarget No matching version found for @eslint/eslintrc@^2.1.4.

文章目录 Intro解决流程总结前置信息了解npm 镜像源三个要用到的npm命令 官方源确认查看当前镜像源的详情解决: 切换镜像源后重试重新操作 事后感受 Intro 事由是今天我在用 create-react-app 新建一个用于测试的前端项目。 然后就出现以下报错: wuyuj…

聚观早报 |国行PS5轻薄版开售;岚图汽车11月交付7006辆

【聚观365】12月2日消息 国行PS5轻薄版开售 岚图汽车11月交付7006辆 比亚迪推出12月限时优惠 特斯拉正式交付首批Cybertruck 昆仑万维发布「天工 SkyAgents」平台 国行PS5轻薄版开售 索尼最新的PlayStation5主机(CFI-2000型号组-轻薄版)国行版本正…

算法通关村第五关—队栈和Hash的经典问题(白银)

emsp;emsp;emsp队栈和Hash的经典问题 用栈实现队列 栈是先进后出,队列是先进先出,所以可以使用两个栈来实现队列的功能。 LeetCode232: 请你仅使用两个栈实现先入先出队列。队列应当支持一般队列支持的所有操作(push、pop、peek、 empty): …

不可抗力因素包括什么内容

一、《民法典》所称的“不可抗力”,是指不能预见、不能避免并不能克服的客观情况。主要包括以下几种情形: (1)自然灾害、如台风、洪水、冰雹; (2)政府行为,如征收、征用; (3)社会异常事件,如罢工、骚乱。 ​…

【设计模式-4.1】行为型——策略模式

说明:本文介绍设计模式中的行为型设计模式中的,策略模式; 计算器 策略模式属于行为型设计模式,关注对象的行为。例如,目前有一个计算器类,可对两个数进行加减计算,如下: &#xf…

Innodb数据空间占用探索

了解数据存储空间占用,可以更方便我们再企业中对于数据库相关优化做评估。 一、查看当前数据表空间占用信息 首先这里准备一张数据库表约2.3w数据量: CREATE TABLE project (tenantsid bigint(20) NOT NULL DEFAULT 0 COMMENT 租户ID,project_id bigi…

Redis RDB

基于内存的 Redis, 数据都是存储在内存中的。 那么如果重启的话, 数据就会丢失。 为了解决这个问题, Redis 提供了 2 种数据持久化的方案: RDB 和 AOF。 RDB 是 Redis 默认的持久化方案。当满足一定条件的时候, 会把当前内存中的数据写入磁盘, 生成一个快照文件 dump.rdb。Redi…

thinkphp 5.1 对数据库查出来的字段进行预处理

比如数据库的设计是下面这样子&#xff1a; 我想展示的是这个样子&#xff1a; 前端可以处理。 Think PHP的处理方式&#xff1a; 定义属性 &#xff1a; $this->customize 任意值;//这里的之没有作用 <?phpnamespace app\hs\controller\shop;use app\daogou\mo…

C++ list容器

文章目录 C++ list容器list基本概念list构造函数list 赋值和交换list 大小操作list 插入和删除list 数据存取list 反转和排序排序案例C++ list容器 list基本概念 功能:将数据进行链式存储 链表(list)是一种物理存储单元上非连续的存储结构,数据元素的逻辑顺序是通过链表中…

ipvlan介绍

最近使用docker&#xff0c;涉及到需要跨多台物理机部署系统&#xff0c;查了好多资料&#xff0c;最后查到了ipvlan。那什么是vlan&#xff0c;什么又是ipvlan。 交换机层面的vlan&#xff0c;是按802.1Q规范&#xff0c;在链路层中加了4字节的标识vlan的数据&#xff0c;交换…