java对象的创建与内存分配机制

news2025/1/13 15:55:45

文章目录

  • 对象的创建与内存分配机制
    • 对象的创建
      • 类加载检查
      • 分配内存
      • 初始化零值
      • 设置对象头
      • 指向init方法
      • 其他:指针压缩
    • 对象内存分配
      • 对象在栈上分配
      • 对象在Eden区中分配
      • 大对象直接分配到老年代
      • 长期存活的对象进入老年代
      • 对象动态年龄判断
      • 老年代空间分配担保机制
    • 对象的内存回收
      • 垃圾标记算法
      • 常见的引用类型
      • finalize()方法
      • 如何判断一个类是无用的类

对象的创建与内存分配机制

对象的创建

java中对象的创建过程如下图所示分为这几步:类加载检查、分配内存、初始化、设置对象头、执行init方法

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-bL9W0Ebb-1678459769746)(picture/性能调优/123315)]



类加载检查

检查当前要创建对象所对应的类是否被加载过,类元数据信息是否已经在方法区中保存过,如果没有则会去进行类加载的过程,如果加载了则进行下一步



分配内存

这一部分需要考虑两点:

  • 应该将哪一块内存区域分配给对象
  • 如何解决多个线程并发分配同一块内存问题

如何分配内存

java中有两种为对象分配内存的方式:

  • 指针碰撞(Bump the Pointer) 默认使用

    对于绝对工整的内存区域,一边是已经分配的内存,一边是未分配的内存,中间有一个指针分割。

    经过内加载后就已经知道了此对象会占用多大的内存,从指针位置开始往后分配一块内存给该对象

  • 空闲列表(Free List)

    对于内存比较碎片化,部署绝对工整的情况,就需要使用空闲列表机制来分配内存了,JVM底层会维护一个空闲列表,然后选出一块合适该对象的内存区域进行分配

解决并发分配问题

在多线程的情况下,就可以会出现多个线程都再指针移动前读取到了当前指针的位置,然后把后面的一块内存区域进行了重复分配。解决这个问题有两种方法:

  • CAS算法

    再使用内存前先进行一次验证,比如判断指针是否移动了,如果验证不通过则进行重试

  • TLAB,JDK8默认使用

    本地线程分配缓冲(Thread Local Allocation Buffer,TLAB)

    为每个线程在Eden区中事先分配一块内存区域,各个线程就在自己的这块区域中进行分配,这样就不会有并发问题,当这块内存区域不够使用时在使用CAS算法在Eden去中分配

    通过-XX:+/-UseTLAB参数来设定虚拟机是否使用TLAB(JVM会默认开启-XX:+UseTLAB),-XX:TLABSize 指定TLAB大小。



初始化零值

为对象的属性赋零值,这里需要注意,在类加载的时候是操作的静态变量,而这里才是操作的普通成员变量

初始化零值的作用是:避免了对象属性没有初始化就被使用



设置对象头

对象头的作用是:保存Hash值、分代年龄、锁信息、指向方法区中的类元信息等等

java的对象一般分为三部分:对象头、实例属性、对其填充

  • 对象头接下来详细讲
  • 实例属性:保存此对象各个属性的值
  • 对其填充:保证整个对象占用内存是8字节的整数倍

对象头如下图所示又分为三个部分

  • mark word标记阶段

    记录对象hash值、分代年龄、锁信息。32位机器占4字节,64为机器占8字节

  • Klass point 类型指针

    指向方法区中类元数据。64位机器开启指针压缩后占4字节,否则占8字节

    虚拟机通过这个指针来确定这个对象是哪个类的实例。

  • 数组长度:只有数组对象才有,占4字节

32位机器中的对象头信息

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-QqDuJNwb-1678459769747)(picture/性能调优/123316)]

64位机器中的对象头信息

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-i4ZEN4f0-1678459769747)(picture/性能调优/123317)]



指向init方法

对成员变量赋程序员自定义的值,并调用对象的构造方法



其他:指针压缩

JDK1.6开始,对于64为的机器默认是开启指针压缩的

启用指针压缩:-XX:+UseCompressedOops(默认开启),禁止指针压缩:-XX:-UseCompressedOops

之所以要使用指针压缩最主要的原因就是节约堆的内存使用



对象内存分配

一个对象的内存分配过程大致如下图所示

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-99q9JK41-1678459769747)(picture/性能调优/123318)]



对象在栈上分配

我们创建一个对象其实最开始的时候会判断该对象是否能够在栈上分配,目的是减轻GC的压力,当方法执行完 出栈后,这一块栈帧内存区域也就被回收了

经过逃逸分析,我们就能判断此对象是否能在栈上分配,其实就是看此对象有没有在该方法之外的地方被引用,是否有作为方法的返回值给其他地方引用。

同时还有标量替换机制,对象是需要一块连续的内存空间,而栈帧的内存可能没有这么大的一块连续的内存空间分配给对象,而标量替换就是将对象拆开,分散存储对象的成员属性。

标量与聚合量,java的基本类型就是标量,java的对象就是聚合量

逃逸分析和标量替换JDK1.7之后都是默认开启的,逃逸分析开关-XX:+DoEscapeAnalysis 标量替换开关-XX:+EliminateAllocations



对象在Eden区中分配

年轻代默认占用堆的1/3内存,其中Eden区大致占用8/10,S0占用1/10, S1占用1/10

JVM参数-XX:+UseAdaptiveSizePolicy(默认开启),会导致这个8:1:1比例自动变化,如果不想这个比例有变化可以设置参数-XX:-UseAdaptiveSizePolicy

一般情况下对象会在Eden区中创建,当存满后进行MinorGC,将存活的对象移至S0区,下一次MinorGC就回收Eden区和S0区,将存活对象移至S1,这其中分代年龄一直递增,默认达到15后就移至老年代

如果一个对象在经过MinorGC后,Survicor的剩余空间不足已存放该对象,这个对象就会提前 直接进入到老年代中

当老年代内存使用满后触发FullGC,回收堆和方法区的内存



大对象直接分配到老年代

JVM的参数 -XX:PretenureSizeThreshold能配置超过这个存储空间的对象就是大对象,直接分配到老年代中,目的是减少大对象来回拷贝占用Survicor的空间

这个参数只在 Serial 和ParNew两个收集器下有效。

所以我们一般都是两个参数一起设置JVM参数:-XX:PretenureSizeThreshold=1000000 (单位是字节) -XX:+UseSerialGC



长期存活的对象进入老年代

对象每经过一次GC,如果还存活该对象年龄则+1,默认达到15次后就进入老年代,CMS收集器默认6岁,不同的垃圾收集器会略微有点不同

对象晋升到老年代的年龄阈值,可以通过参数 -XX:MaxTenuringThreshold 来设



对象动态年龄判断

经过一次MinorGC后,存活的对象会存放在Survicor区,这一批对象总大小如果大于了当前Servicor区内存的一半,此时对象年龄大于等于这批对象最大年龄的对象就直接进入到老年代,比如此时Servicor区现有一些对象,年龄1+年龄2+年龄n的多个年龄对象总和超过了Survivor区域的50%,此时就会把年龄n(含)以上的对象都放入老年代

-XX:TargetSurvivorRatio 目标存活率,默认为50%

对象动态年龄判断机制一般是在minor gc之后触发的。



老年代空间分配担保机制

每经过一个MinorGC前都会计算当前老年代剩余可用空间,判断老年代剩余可用空间是否小于年轻代所有对象总大小

如果小于则检查当前是否有-XX:-HandlePromotionFailure参数,JDK8默认有

如果有这个参数则判断当前老年代剩余可用空间是否小于以往MinorGC移入老年代对象内存空间的平均值

如果小于或者上一步参数没有配置则直接进行FullGC,之后其实还是会进行一次MinorGC

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-c1BrSxtg-1678459769747)(picture/性能调优/123319)]



对象的内存回收

垃圾标记算法

  • 引用计数法,对象循环引用问题无法解决
  • 可达性分析算法,一般默认使用的这种算法,从GCRoot对象开始向下寻找引用。GFRoot对象是栈帧的成员变量、静态变量



常见的引用类型

  • 强引用

    普通的变量引用,GC不会被清理

  • 软引用

    将对象用SoftReference对象包裹,正常情况下不会被回收,如果经过一次GC后还是释放不出空间存放新的对象则会将软引用的对象回收

    public static SoftReference<User> user = new SoftReference<User>(new User());
    
  • 弱引用

    将对象用WeakReference软引用类型的对象包裹,弱引用跟没引用差不多,GC会直接回收掉,很少用

    public static WeakReference<User> user = new WeakReference<User>(new User());
    
  • 虚引用

    虚引用也称为幽灵引用或者幻影引用,它是最弱的一种引用关系,几乎不用



finalize()方法

finalize()方法最终判定对象是否存活,当一个对象被标记为垃圾对象后,如果该对象重写了finalize()方法则不会立刻回收该对象,此时该对象有一次自救的机会,只要重新与引用链上的任何的一个对象建立关联即可。那在第二次标记时它将移除出“即将回收”的集合。如果对象这时候还没逃脱,那基本上它就真的被回收了。

注意:一个对象的finalize()方法只会被执行一次,也就是说通过调用finalize方法自我救命的机会就一次。

finalize()方法的运行代价高昂, 不确定性大, 无法保证各个对象的调用顺序, 如今已被官方明确声明为不推荐使用的语法。



如何判断一个类是无用的类

在进行FullGC回收方法区中的内存时,就需要判断类是不是无用的类,

需要满足下面三个条件才是无用的类:

  • 类的实例对象都已经被回收了
  • 加载该类的类加载器被回收了
  • 堆中该类的java.lang.Class 对象没有在任何地方被引用

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

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

相关文章

Spring的核心模块:Bean的生命周期(内含依赖循环+业务场景)。

Bean的生命周期前言为什么要学习Bean的生命周期前置知识Spring Post-processor&#xff08;后置处理器&#xff09;Aware接口简单介绍Bean的实例化过程为什么会有bean的实例化&#xff1f;过程Bean的初始化阶段为什么会有Bean的初始化&#xff1f;Bean的初始化目的是什么&#…

线性和非线性最小二乘问题的常见解法总结

线性和非线性最小二乘问题的各种解法 先看这篇博客&#xff0c;非常好&#xff1a;线性和非线性最小二乘问题的各种解法 1. 线性最小二乘问题有最优解 但是面对大型稀疏矩阵的时候使用迭代法效率更好。 迭代法 有Jacobi迭代法、 Seidel迭代法及Sor法 【数值分析】Jacobi、Se…

[ubuntu][GCC]gcc源码编译

1.下载gcc安装包 https://ftp.gnu.org/gnu/gcc/ 选择一个需要的gcc版本&#xff0c;下载。 2.下载依赖包 查看下载的gcc安装包中contrib文件夹下的download_prerequisites文件&#xff0c;查看需要的依赖包版本。 根据download_prerequisites中红框位置的信息&#xff0c;在下…

JSON.stringify()的5种使用场景

JSON.stringify() 方法将一个JavaScript对象或值转换为JSON字符串&#xff0c;如果指定了一个replacer函数&#xff0c;则可以选择性地替换值&#xff0c;或者指定的replacer是数组&#xff0c;则可选择性地仅包含数组指定的属性。 语法如下&#xff1a; JSON.stringify(value…

电子技术课程设计基于FPGA的音乐硬件演奏电路的设计与实现

wx供重浩&#xff1a;创享日记 对话框发送&#xff1a;乐曲电路 免费获取完整无水印论文报告&#xff08;包含电路图&#xff09; 文章目录一、设计任务要求二、总体框图三、选择器件四、功能模块五、总体设计电路图六、结束语一、设计任务要求 1、课程设计题目 设计一个乐曲演…

【Flutter从入门到入坑之三】Flutter 是如何工作的

【Flutter从入门到入坑之一】Flutter 介绍及安装使用 【Flutter从入门到入坑之二】Dart语言基础概述 【Flutter从入门到入坑之三】Flutter 是如何工作的 本文章主要以界面渲染过程为例&#xff0c;介绍一下 Flutter 是如何工作的。 页面中的各界面元素&#xff08;Widget&…

使数组和能被P整除[同余定理+同余定理变形]

同余定理同余定理变形前言一、使数组和能被P整除二、同余定理变形总结参考资料前言 同余定理非常经典&#xff0c;采用前缀和 map&#xff0c;当两个余数前缀和为一个值时&#xff0c;则中间一段子数组刚好对P整除。但是能否找到前面是否有一段子数组和可以对P整除呐&#xf…

认识CSS之元素显示模式

&#x1f31f;所属专栏&#xff1a;前端只因变凤凰之路&#x1f414;作者简介&#xff1a;rchjr——五带信管菜只因一枚&#x1f62e;前言&#xff1a;该系列将持续更新前端的相关学习笔记&#xff0c;欢迎和我一样的小白订阅&#xff0c;一起学习共同进步~&#x1f449;文章简…

深度学习训练营之数据增强

深度学习训练营学习内容原文链接环境介绍前置工作设置GPU加载数据创建测试集数据类型查看以及数据归一化数据增强操作使用嵌入model的方法进行数据增强模型训练结果可视化自定义数据增强查看数据增强后的图片学习内容 在深度学习当中,由于准备数据集本身是一件十分复杂的过程,…

Python 中 KeyError: 0 exception 错误

Python “KeyError: 0” 异常是在我们尝试访问不包含0 这个键的时候去访问该键而引起的。 要解决该错误&#xff0c;请在尝试访问字典之前在字典中设置键&#xff0c;或者如果键不存在&#xff0c;则使用 dict.get() 获取默认值。 下面是一个产生上述错误的示例 my_dict {1…

KDZD互感器二次负载测试仪

一、概述 电能计量综合误差过大是电能计量中普遍存在的一个关键问题。电压互感器二次回路压降引起的计量误差往往是影响电能计量综合误差的因素。所谓电压互感器二次压降引起的误差&#xff0c;就是指电压互感器二次端子和负载端子之间电压的幅值差相对于二次实际电压的百分数…

五分钟了解JumpServer V2.* 与 v3 的区别

一、升级注意项 1、梳理数据。JumpServer V3 去除了系统用户功能&#xff0c;将资产与资产直接绑定。当一个资产名下有多个同名账号&#xff0c;例如两个root用户时&#xff0c;升级后会自动合并最后一个root&#xff0c;不会同步其他root用户。升级前需保证每一个资产只拥有一…

即时通讯系列-N-客户端如何在推拉结合的模式下保证消息的可靠性展示

结论先行 原则&#xff1a; server拉取的消息一定是连续的原则&#xff1a; 端侧记录的消息的连续段有两个作用&#xff1a; 1. 记录消息的连续性&#xff0c; 即起始中间没有断层&#xff0c; 2. 消息连续&#xff0c; 同时意味着消息是最新的&#xff0c;消息不是过期的。同…

Java学习-MySQL-创建数据库表

Java学习-MySQL-创建数据库表 SHOW DATABASESUSE school CREATE TABLE IF NOT EXISTS student( id INT(10) NOT NULL AUTO_INCREMENT COMMENT 学号, name VARCHAR(30) NOT NULL DEFAULT 匿名 COMMENT 姓名, pws VARCHAR(20) NOT NULL DEFAULT 123456 COMMENT 密码, sex VARCHA…

算法题--二叉树(判断是不是平衡二叉树、二叉树的中序遍历、二叉树最大深度、对称二叉树、合并二叉树)

目录 二叉树 题目 判断是不是平衡二叉树 题链接 解析 核心思想 答案 二叉树的中序遍历 原题链接 解析 核心思想 答案 二叉树最大深度、对称二叉树、合并二叉树 二叉树 该类题目的解决一般是通过节点的遍历去实现&#xff0c;一般是分两种。 一是递归&#xff08;…

【记录】日常|shandianchengzi的三周年创作纪念日

机缘 接触 CSDN 之前&#xff0c;我已经倒腾过 hexo 搭建 github 博客、本地博客、图床&#xff1b;   接触 CSDN 之后&#xff0c;我还倒腾过纸质笔记、gitee 博客、博客园、知乎、b站、Notion、腾讯文档、有道云笔记、XMind、飞书文档、简书等一系列创作平台&#xff0c;但…

SAPUI5开发01_01-Installing Eclipse

1.0 简要要求概述: 本节您将安装SAPUI 5,以及如何在Eclipse Juno中集成SAPUI 5工具。 1.1 安装JDK JDK 是一种用于构建在 Java 平台上发布的应用程序、Applet 和组件的开发环境,即编写 Java 程序必须使用 JDK,它提供了编译和运行 Java 程序的环境。 在安装 JDK 之前,首…

1635_fileno的简单使用

全部学习汇总&#xff1a; GreyZhang/g_unix: some basic learning about unix operating system. (github.com) 在看MIT的OS课程的时候发现自己动不动就因为只是的缺少而卡住&#xff0c;而这个学习占据了我工作之余很多的时间。现在都有一点觉得通关不了的感觉了&#xff0c;…

1. Qt Designer Studio界面介绍

1. 说明&#xff1a; Qt当中的Qt Quick框架使用QML语言来快速搭建优美的界面&#xff0c;但是对于单纯做界面的设计人员并不是很友好&#xff0c;还要让界面设计人员去消耗时间成本学习QML语法。Qt Designer Studio软件就是为了解决这个问题而设计的&#xff0c;工作人员不需要…

【Blender】Stability AI插件 - AI生成图像和动画

Stability AI 的官方插件允许 Blender 艺术家使用现有的项目和文本描述来创建新的图像、纹理和动画。 推荐&#xff1a;用 NSDT场景设计器 快速搭建3D场景。 1、安装Stability for Blender插件 首先&#xff0c;从这里下载最新版本的 Blender&#xff0c;然后转到 Addon Relea…