Java内存结构

1.对象的结构

一个Java对象在内存中包括3个部分:对象头、实例数据和对齐填充

2.虚拟机存储数据的方式

2.1小端存储 :

便于数据之间的类型转换,例如:long类型转换为int类型时,高地址部分的数据可以直接截掉。

2.2大端存储 :

便于数据类型的符号判断,因为最低地址位数据即为符号位,可以直接判断数据的正负号。(java用的)

假设存储一个变量:int i = 1; 

3.访问到对象的2种方式 

内存模型设计之–Class Pointer

3.1句柄池访问:

使用句柄访问对象,会在堆中开辟一块内存作为句柄池,句柄中储存了对象实例数据(属性值结构体) 的内存地址,访问类型数据的内存地址(类信息,方法类型信息),对象实例数据一般也在heap中开 辟,类型数据一般储存在方法区中。

优点 :

reference存储的是稳定的句柄地址,在对象被移动(垃圾收集时移动对象是非常普遍的行为) 时只会改变句柄中的实例数据指针,而reference本身不需要改变。

缺点 :

增加了一次指针定位的时间开销。

3.2直接指针访问对象:(Java选用)

直接指针访问方式指reference中直接储存对象在heap中的内存地址,但对应的类型数据访问地址需要 在实例中存储。

优点:

节省了一次指针定位的开销。

缺点 :

在对象被移动时(如进行GC后的内存重新排列),reference本身需要被修改

4.内存模型设计之–指针压缩技术

4.1指针压缩的目的:

1. 为了保证CPU普通对象指针(oop)缓存
2. 为了减少GC的发生,因为指针不压缩是8字节,这样在64位操作系统的堆上其他资源空间就少了。

64操作系统与32位操作系统的区别:

32位的处理器

一次能够去处理32个二进制位 ,也就是一次处理4字节的数据 ,2的32次方的寻址空间(4G)

64位操作系统  

 一次能够去处理64个二进制位 ,也就是一次处理4字节的数据,2的64次方的寻址空间(16G)

4.2指针压缩失效的原因

64位操作系统中内存大于4G 默认开启指针压缩技术,内存小于4G,默认是32位系统默认不启内存大于32G指针压缩失效。所以我们通常在部署服务时,JVM内存不要超过32G,因为超过32G就无法开启指针压缩了。

4.2.1失效原因

32位系统的CPU 最大支持2^32 = 4G ,如果是64位系统,最大支持 2^64, 但是对其填充是按照8字节进行填充,指针压缩可以理解为在32位系统在64位上面使用,因为32位系统的CPU寻址空间最大支持4G,但是有8字节的对其填充所以要乘以8 = 32G,这就是内存大于32G指针压缩失效的原因。

关闭指针压缩 : -XX:-UseCompressedOops

如果现在老项目    32位操作系统  支持    4G以上的内存

PAE的特殊内核

4.3内存模型设计之–对齐填充

对齐填充的意义是提高CPU访问数据的效率,主要针对会存在该实例对象数据跨内存地址区域存储的情况。

例如:在没有对齐填充的情况下,内存地址存放情况如下:

因为处理器只能0x00-0x07的8位数据,0x08-0x0F这样读取数据,所以当我们想获取这个long型的数据时,处理器必须要读两次内存,第一次(0x00-0x07),第二次(0x08-0x0F),然后将两次的结果汇总才能获得真正的数值。

那么在有对齐填充的情况下,内存地址存放情况是这样的:

0x07的位置不存储数据,现在处理器只需要直接一次读取(0x08-0x0F)的内存地址就可以获得我们想要的数据了。

当下次有个boolean类型在进来可以直接填充到0x07中

4.3.1Java类在读取到内存时并不是安装顺序来的

可以配置参数

当我们的策略为0时,这个时候我们的加载顺序是  基本类型>填充字段>引用类型

当我们策略为1时,这个时候我们的加载顺序是,引用类型>基本类型>填充字段

策略为2时,父类中的引用类型跟子类中的引用类型放在一起,其他类型父类采用策略0,子类采用策略1,这样操作可以降低空间的开销 

当一个项目中继承关系特别多的时候就可以使用策略2,由于引用类型都放在一起,GC一次就可以回收所有的父子类引用类型。

4.3.2如果父类的内存空间由于8字节的对其,导致中间有空位,子类是否可以填充?

父类的内存有空的话不可以存放子类的数据,因为父子类有内存隔离,假设放入子类数据,需要记住这个数据时子类的还是父类的,而且基本类型在GC的时候会被回收,导致子类没有数据

5. 运行时数据区

上面对运行时数据区描述了很多,其实重点存储数据的是堆和方法区(非堆),所以内存的设计也着重从这两方面展开(注意这两块区域都是线程共享的)。

对于虚拟机栈,本地方法栈,程序计数器都是线程私有的

可以这样理解,JVM运行时数据区是一种规范,而JVM内存模式是对该规范的实现

6.JVM内存模型

一块是非堆区,一块是堆区
堆区分为两大块,一个是Old区,一个是Young区
Young区分为两大块,一个是Survivor区(S0+S1),一块是Eden区
S0和S1一样大,也可以叫From和To 

6.1GC 

GC的流程

如何理解Minor/Major/Full GC

Minor GC:新生代
Major GC:老年代
Full GC:新生代+老年代

对象创建过程

一般情况下,新创建的对象都会被分配到Eden区,一些特殊的大的对象会直接分配到Old区。

我是一个普通的Java对象,我出生在Eden区,在Eden区我还看到和我长的很像的小兄弟,我们在Eden区中玩了挺长时间。有一天Eden区中的人实在是太多了,我就被迫去了Survivor区的“From”区,自从去了Survivor区,我就开始漂了,有时候在Survivor的“From”区,有时候在Survivor的“To”区,居无定所。直到我18岁的时候,爸爸说我成人了,该去社会上闯闯了。于是我就去了年老代那边,年老代里,人很多,并且年龄都挺大的。

6.1.2区分新生代与老年代的标准 

每一次GC都会对对像的年龄进行加一,当age=15时就会进入老年代,在对象的MarkWord中会存放对象的年龄,存储这些信息的空间只有4个2进制位,也就是(0000-1111)

6.1.2.1那么为什么是年龄到15才进入老年代?

进入old区的对象越少越好,也就是age越大越好,但是根据IBM的实验数据统计,98%的对象都是活不过一次GC的,那么剩下的也就是2%,15次GC最多存活的数据也就是30%,而且2%的对象也有活不过2次GC的,综合下来15次GC活下来的也就是10%。

7.JVM承认GC的类型

7.1Partial GC 部分GC

Minor GC:新生代
Major GC:老年代

7.2Full GC 全局GC

Full GC:Young GC+Old GC+MateSpace GC

7.3什么时候会触发Full   GC?

1.之前每次晋升的对象的平均大小 >老年代的剩余空间,基于历史平均水平,本次GC后要到老年代的数据大于前面的平均大小

2.young  GC之后 ,存活对象超过了老年代的剩余空间 ,基于下一次可能的剩余空间

3.Meta Space区域空间不足

4.System.gc();

7.4Meta Space空间

JDK1.7之前是Perm  space 也叫永久代 ,持久代 ,使用的是JVM自己的内存   线性整理    会增加垃圾回收的时间

设置初始化堆内存为2G,不设置的方法区的内存大小,当内存不够时会直接溢出

 给方法区设置500M的内存空间,是包含在整个Jvm内存中的,初始化堆内存即便设置1.8G,实际的内存也只会有1.5G,因为方法区的优先级高于初始化堆内存,这样1.5G就不够用,会导致启动项目直接FullGC

JDK1.8    Meta Space是方法区的实现 也叫元空间,元数据区       直接内存   减少内存碎片        节省压缩时间

MateSpace只把静态变量和字符串常量放到初始化堆内存中,其他的全部放入服务器内存中,而且当自己内存不够时就会动态扩容(权限高)

 8.GC的悲观策略

在某些情况下可以不经过GC直接进入老年代

1.当S区中相同年龄和大于这个年龄的所有对象大小,大于S区中任何一个的一半时会直接进入老年代

2.当Eden区的所有对象经过GC后,后面进来的对象大小依然无法进入Eden区时也会直接进入老年代

9.为什么需要Survivor区?只有Eden不行吗?

如果没有Survivor,Eden区每进行一次Minor GC,存活的对象就会被送到老年代。这样一来,老年代很快被填满,触发Major GC(因为Major GC一般伴随着Minor GC,也可以看做触发了Full GC)。
老年代的内存空间远大于新生代,进行一次Full GC消耗的时间比Minor GC长得多。
执行时间长有什么坏处?

频发的Full GC消耗的时间很长,会影响大型程序的执行和响应速度。

可能你会说,那就对老年代的空间进行增加或者较少咯。
假如增加老年代空间,更多存活对象才能填满老年代。虽然降低Full GC频率,但是随着老年代空间加大,一旦发生Full GC,执行所需要的时间更长。
假如减少老年代空间,虽然Full GC所需时间减少,但是老年代很快被存活对象填满,Full GC频率增加。

所以Survivor的存在意义,就是减少被送到老年代的对象,进而减少Full GC的发生,Survivor的预筛选保证,只有经历16次Minor GC还能在新生代中存活的对象,才会被送到老年代。

10.为什么需要两个Survivor区?

最大的好处就是解决了碎片化。也就是说为什么一个Survivor区不行?第一部分中,我们知道了必须设置Survivor区。假设现在只有一个Survivor区,我们来模拟一下流程:
刚刚新建的对象在Eden中,一旦Eden满了,触发一次Minor GC,Eden中的存活对象就会被移动到Survivor区。这样继续循环下去,下一次Eden满了的时候,问题来了,此时进行Minor GC,Eden和Survivor各有一些存活对象,如果此时把Eden区的存活对象硬放到Survivor区,很明显这两部分对象所占有的内存是不连续的,也就导致了内存碎片化。
永远有一个Survivor space是空的,另一个非空的Survivor space无碎片。

 Eden区经过一次GC后,剩下两个对象,但是这两个对象在内存中不是连续的,这时要进来一个70M的对象,就会造成明明Eden有空间但是放不进去,这时就会放到survivor区,这样70M的对象就可以进入Eden区

如果进行一次GC是这样情况

S0 区GC后造成了2个内存不连续的对象,而且Eden中也有这种情况,这时有个70M对象要进Eden区,这时Eden区有空间却进不去,而且S0区也是有空间,但是无法接收Eden中的两个对象,这时S1就会接收S0的对象,这样70M的对象就可以进入Eden区

11.新生代中Eden:S1:S2为什么是8:1:1?

新生代中的可用内存:复制算法用来担保的内存为9:1
可用内存中Eden:S1区为8:1
即新生代中Eden:S1:S2 = 8:1:1
现代的商业虚拟机都采用这种收集算法来回收新生代,IBM公司的专门研究表明,新生代中的对象大概98%是“朝生夕死”的

12.堆内存中都是线程共享的区域吗?

JVM默认为每个线程在Eden上开辟一个buffer区域,用来加速对象的分配,称之为TLAB,全称:Thread Local Allocation Buffer。
对象优先会在TLAB上分配,但是TLAB空间通常会比较小,如果对象比较大,那么还是在共享区域分配。

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

德迅云安全告诉您 网站被攻击怎么办-SCDN来帮您

随着互联网的发展给我们带来极大的便利,但是同时也带来一定的安全威胁,网络恶意攻击逐渐增多,很多网站饱受困扰,而其中最为常见的恶意攻击就是cc以及ddos攻击。针对网站攻击,今天为您介绍其中一种防护方式-SCDN&#x…

Zabbix实现故障自愈

一、简介 Zabbix agent 可以运行被动检查和主动检查。 在被动检查模式中 agent 应答数据请求。Zabbix server(或 proxy)询求数据,例如 CPU load,然后 Zabbix agent 返还结果。 主动检查处理过程将相对复杂。Agent 必须首先从 Z…

【算法每日一练]-分块(保姆级教程 篇1)POJ3648

插讲一下分块 题目:(POJ 3648) 一个简单的整数问题 前缀和往往用于静态的不会修改的区间和。遇到经常修改的区间问题,就要用分块或线段树来维护了。 分块算法是优化后的暴力,分块算法有时可以维护一些线段树维护不了的…

Michael Jordan最新报告:去中心化机器学习中的契约、不确定性和激励

‍ ‍导读 11月3日,智源研究院学术顾问委员会委员、机器学习泰斗Michael Jordan在以“新一代人工智能前沿”为主题的2023北京论坛 新工科专题论坛上,发表了题为Contracts, Uncertainty, and Incentives in Decentralized Machine Learning(去…

RHCSA --- Linux存储管理

存储管理 Boot:可引导操作系统的分区(必须是主分区) 分区 ll /dev/nvme0n* 表示的是 nvme接口的磁盘 0n1 1 0n2 2 0n3 3 brw-rw----. 1 root disk 259, 0 Nov 15 19:31 /dev/nvme0n1 磁盘1 brw-rw----. 1 ro…

「Verilog学习笔记」根据状态转移图实现时序电路

专栏前言 本专栏的内容主要是记录本人学习Verilog过程中的一些知识点,刷题网站用的是牛客网 这是一个典型的米利型状态机。三段式即可解决。 米利型状态机:即输出不仅和当前状态有关,也和输入有关。 其中ST0,ST1,ST3的…

FastJson竟然会导致内存泄露?你遇到过吗?

FastJson是一款性能优异的java序列化和反序列框架,广泛应用于日常开发工作中,也许正是因为作者在设计这款框架时,比较注重性能方面的考量,在框架安全性,空间占用等方面做了一些牺牲。 很不幸小编前两天就遇到了一个使…

文件重命名最佳实践:如何确保文件名的准确性和一致性

在日常生活和工作中,经常需要处理大量的文件,包括电子文件和纸质文件。文件重命名是为了更好地组织和管理这些文件,以方便查找和使用。然而,重命名文件并不是一件简单的事情,它需要遵循一定的最佳实践以确保文件名的准…

视频封面:从视频中提取封面,轻松制作吸引人的视频

在当今的数字时代,视频已成为人们获取信息、娱乐和交流的重要方式。一个吸引人的视频封面往往能抓住眼球,提高点击率和观看率。今天将介绍如何从视频中提取封面,轻松制作吸引人的视频封面。 一、准备素材选择合适的视频片段 首先&#xff0…

母婴服务预约小程序的效果如何

二胎家庭增速明显,占比较大,成为市场各母婴品牌的目标,而随着行业发展及市场变化,线上互联网深入人们生活,各家母婴品牌开始向“数字化”靠拢。 目前母婴门店商家主要面临服务/产品线上曝光不足、宣传度不够或扩圈无门…

JAVA小游戏 “拼图”

第一步是创建项目 项目名自拟 第二部创建个包名 来规范class 然后是创建类 创建一个代码类 和一个运行类 代码如下: package heima;import java.awt.event.ActionEvent; import java.awt.event.ActionListener; import java.awt.event.KeyEvent; import jav…

EDA实验-----4*4矩阵键盘与数码管显示测试

目录 一、实验目的 二、实验仪器设备 三、实验原理 四、实验要求 五、实验步骤 六、实验报告 七、实验过程 1.矩阵键盘按键原理 2.数码管原理 3.分频器代码 4.电路图连接 5.文件烧录 一、实验目的 了解数码管的工作原理;掌握4*4矩阵键盘和数码管显示的编…

UnitTest框架

目标: 1.掌握UnitTest框架的基本使用方法 2.掌握断言的使用方法 3.掌握如何实现参数化 4.掌握测试报告的生成 1.定义 (1)框架(framework):为解决一类事情的功能集合。(需要按照框架的规定(套路) 去书写代码&…

Virtual安装centos后,xshell连接centos

1. 网络使用Host-Only模式动态分配IP,运行 system restart network 后,使用ifconfig查看新的ip,XShell可以直接连上centos, 但是由于使用的是Host-Only模式,centos不能访问网络,只能与宿主机相互通信 2. 网…

快速支持客户知识库的核心优势是什么?

快速支持客户知识库是一个集中存储和组织企业知识的平台,包含了丰富的信息和解决方案,以帮助客户快速解决问题,帮助企业提高客户支持效率和满意度。那么,快速支持客户知识库的核心优势是什么呢? | 1、提高客户自助支持…

VBA之Word应用:文档(Document)的书签

《VBA之Word应用》(版权10178982),是我推出第八套教程,教程是专门讲解VBA在Word中的应用,围绕“面向对象编程”讲解,首先让大家认识Word中VBA的对象,以及对象的属性、方法,然后通过实…

win10无损升级到win11

1,下载win11升级助手 https://download.microsoft.com/download/5/4/c/54c22b82-d0cd-4e34-9a06-b75823a8aede/Windows11InstallationAssistant.exe 2,启动助手开始安装 安装时需要重启数次 3,安装后界面 4,安装后&#xff0c…

ACWSpring1.3

首先,前端写ajax写上我们的访问路径(就在我们前端的源代码里面),我们建了两个包pkController用于前端页面url映射过来一层一层找到我们的RestController返回bot1里面有键值,返回的这就是一个session对象bot1这个map.前端拿到我们bot1里的两个值给到我们前端显示出来 1准备页面:…

苹果签名应用掉签频繁原因排查,以及如何避免

作为一个对iOS生态有着深厚理解的实用技术博主,我明白苹果签名应用掉签对我们的开发和使用带来的困扰。签名在苹果设备中扮演着至关重要的角色,它不仅确保了应用来源的合法性,也影响着应用的顺畅运行。 今天,我将和您一同探讨苹果…

云存储与物理存储:优缺点对比分析

当您需要存储数字文件时,您有两个基本选择:云存储和物理存储。 云存储允许您通过互联网将文件保存在云存储提供商运营的服务器上。这些公司通常在多个数据中心制作文件的备份副本,并使用复杂的加密来保护它们。您可以从任何连接互联网的设备访…
最新文章