JVM运行时数据区域

文章目录

  • 内存结构
      • 程序计数器(寄存器)
      • 虚拟机栈
        • 局部变量表
          • 两类异常状况
        • 线程运行诊断
      • 本地方法栈
      • 方法区
      • 运行时常量池
        • 串池(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 arrayreturn 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.mzph.cn/news/191358.shtml

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

相关文章

(三)Tiki-taka算法(TTA)求解无人机三维路径规划研究(MATLAB)

一、无人机模型简介: 单个无人机三维路径规划问题及其建模_IT猿手的博客-CSDN博客 参考文献: [1]胡观凯,钟建华,李永正,黎万洪.基于IPSO-GA算法的无人机三维路径规划[J].现代电子技术,2023,46(07):115-120 二、Tiki-taka算法(TTA&#xf…

Overflow Aware Quantization

Overflow Aware Quantization Framework N o _o o​是 amount of arithmetic overflow 辅助信息 作者未提供代码

A--Z与a--z的ASCII码的差异

从z到A还有一些字符 应该改为str[i]>A&&str[i]<Z||str[i]>a&&str[i]<z;

架构图是什么,怎么做?

架构图是一种用来描述系统或软件的结构和组成的图形表示。它展示了系统中各个组件之间的关系、交互和功能。通过绘制架构图&#xff0c;可以更好地理解和沟通系统的设计和实现。 绘制架构图的软件 目前市场上有许多用于绘制架构图的软件工具&#xff0c;下面简单…

软件工程 - 第8章 面向对象建模 - 4 - 物理体系结构建模

构件图 构件图概述 构件图描述了软件的各种构件和它们之间的依赖关系。 构件图的作用 在构件图中&#xff0c;系统中的每个物理构件都使用构件符号来表示&#xff0c;通常&#xff0c;构件图看起来像是构件图标的集合&#xff0c;这些图标代表系统中的物理部件&#xff0c;…

pandas美化表格并导出到Excel

美化Excel表格用两种方式,一种是用Pandas自带的Dataframe.style类并通过CSS来改变样式,另外一种是通过Excel引擎来直接修改Excel样式。 Dataframe.style Dataframe.style可以美化Pandas样式。导出样式到Excel的功能只有openpyxl渲染引擎支持。 大于平均数的单元格背景变色…

JavaScript 延迟加载的艺术:按需加载的最佳实践

&#x1f90d; 前端开发工程师&#xff08;主业&#xff09;、技术博主&#xff08;副业&#xff09;、已过CET6 &#x1f368; 阿珊和她的猫_CSDN个人主页 &#x1f560; 牛客高级专题作者、在牛客打造高质量专栏《前端面试必备》 &#x1f35a; 蓝桥云课签约作者、已在蓝桥云…

React Native 更换淘宝镜像提升包下载速度

React Native 更换淘宝镜像提升包下载速度 每次运行项目的时候都是卡在包下载的命令上&#xff0c;每次一等就要 1h20m 极度崩溃&#xff0c;那是因maven镜像源为Google导致无法正常下载。 那么我们就可以切换maven镜像源&#xff0c;方法如下&#xff1a; 找到项目下的**/an…

逻辑回归与正则化 逻辑回归、激活函数及其代价函数

逻辑回归、激活函数及其代价函数 线性回归的可行性 对分类算法&#xff0c;其输出结果y只有两种结果{0&#xff0c;1}&#xff0c;分别表示负类和正类&#xff0c;代表没有目标和有目标。 在这种情况下&#xff0c;如果用传统的方法以线性拟合 &#xff08; h θ ( x ) θ T…

医疗器械设备模组的具体应用

直线模组是一种高精度、高速度的精密传动元件&#xff0c;目前被广泛应用在各种工业自动化领域&#xff1b;尤其是在激光加工、电子制造、医疗设备、物流设备和机器人等行业中&#xff0c;都发挥着重要作用&#xff0c;接下来我们看看医疗器械设备模组的具体应用吧&#xff01;…

多模块下MyBatis官方生成器

MyBatis官方生成器 1. 依赖插件 创建generator模块 在父模块中进行版本管理 <dependency><groupId>org.mybatis</groupId><artifactId>mybatis</artifactId><version>3.5.6</version> </dependency><dependency><g…

Swin Transformer实战图像分类(Windows下,无需用到Conda,亲测有效)

目录 前言 一、从官网拿到源码&#xff0c;然后配置自己缺少的环境。 针对可能遇到的错误&#xff1a; 二、数据集获取与处理 2.1 数据集下载 2.2 数据集处理 三、下载预训练权重 四、修改部分参数配置 4.1 修改config.py 4.2 修改build.py 4.3 修改units.py 4.4 修…

Linux 基础认识

文章目录 前言Linux历史window历史Linux地位发行版本 前言 建议只看概述 Linux历史 概述&#xff1a; 由一个研究生受Minix操作系统启发编写的&#xff0c;因为功能实用&#xff0c;代码开源被世界人接收和开发 &#xff0c;最终正式发布 。 详情&#xff1a; 1991年10月5日…

模式识别与机器学习(七):集成学习

集成学习 1.概念1.1 类型1.2 集成策略1.3 优势 2. 代码实例2.1boosting2.2 bagging2.3 集成 1.概念 集成学习是一种机器学习方法&#xff0c;旨在通过组合多个个体学习器的预测结果来提高整体的预测性能。它通过将多个弱学习器&#xff08;个体学习器&#xff09;组合成一个强学…

docker安装卸载使用全教程

docker安装卸载使用全教程 1.确定你是Centos7及以上版本 cat /etc/redhat-release2.yum安装gcc相关依赖 yum install -y gcc Yum install -y gcc-c3.安装需要的软件包 sudo yum install -y yum-utils4.设置stable镜像仓库 这里设置的是国内阿里云的镜像库&#xff0c;国外的…

SQL Server 2016(为数据表Porducts添加数据)

1、实验环境。 某公司有一台已经安装了SQL Server 2016的服务器&#xff0c;并已经创建了数据库PM。 2、需求描述。 在数据库PM中创建表products&#xff0c;"编号"列的值自动增长并为主键。然后使用T-SQL语句为表格插入如下数据。 3、实验步骤。 1、使用SSMS管理工…

欧洲各国及发达国家经济支柱和第一出口商品是什么

工业在欧洲各国经济支柱中的表现 一般发达国家&#xff0c;像西欧的国家第三产业即服务业占GDP70%甚至更高&#xff0c;从业人数比重也最大&#xff0c;只是越发达的国家服务业的知识性和科技含量会更高&#xff0c;如商业咨询、律师、医疗卫生、科技服务、商业服务。服务业的…

PVE系列-CT容器安装openwrt X86的极简方法

下载推荐&#xff1a;https://openwrt.ai/ 使用环境PVE8.0&#xff0c;openwrt是以上网址的最新版&#xff0c;内涵及其丰富组件。 问题来源&#xff1a; 在PVE虚拟机可以很方便的使用img文件&#xff0c;转换qm 成一个硬盘文件&#xff0c;加入到虚拟机也就完成了&#xff0c…

jQuery的使用

目录 jquery对象&#xff1a; jquery作为一般函数调用参数: jquery事件机制 jquery dom操作 jquery对象&#xff1a; <!DOCTYPE html> <html lang"en"> <head><meta charset"UTF-8"><meta name"viewport" cont…

Linux下配置邮箱客户端MUTT,整合msmtp + procmail + fetchmail

一、背景 在向 Linux kernel 社区提交patch补丁步骤总结&#xff08;已验证成功&#xff09;_kernel补丁-CSDN博客文章中提到如何向kernel社区以及其他类似如qemu、libvirt社区提交patch的详细步骤&#xff0c;但还有一点不足的是通过git send-email这种方法基本是只能发送patc…