热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

jvm对象从新生代到老年代_JVM内存模型2(堆)

4.堆一个JVM实例只存在一个堆内存,堆内存是java内存管理的核心区域,在JVM启动的时候就被创建了,大小也确定了(可调节)堆在物理上可

4. 堆

一个JVM实例只存在一个堆内存,堆内存是java内存管理的核心区域,在JVM启动的时候就被创建了,大小也确定了(可调节)

堆在物理上可以不连续,但是逻辑上应被视为连续的

方法结束后,堆中的对象不会立即消失,要等到gc后才会消失

进程的所有线程共享Java堆,但是还可以划分线程私有的缓冲区:TLAB (Thread Local Allocation Buffer)

  • jdk1.7及之前,堆分为:新生代(Eden:Survivor0:Survivor1=8:1:1),老年代。(永久代属于方法区)
  • jdk1.8及之后,堆分为:新生代,老年代。(元空间属于方法区)
1c6624fbde8ae64a67ee721eade9c790.png

设置堆的大小

设置堆区起始内存大小:-Xms或者-XX:InitialHeapSize,默认大小是:电脑物理内存大小/64

设置堆区最大内存大小:-Xmx或者-XX:MaxHeapSize,默认大小是:电脑物理内存大小/4

设置起始堆大小是只包含新生代和老年代的,不包含永久代/元空间

一旦堆区内存大小超过设置的最大内存大小,就会出现OOM

通常将起始内存和最大内存设置相同的值,不需要在gc清理完堆后重新计算调整堆区大小,从而提高性能

查看堆大小相关命令:

jps:查看java进程的pid

jstat -gc pid:查看pid的内存信息

或者直接添加运行参数-XX:+PrintGCDetails,在程序执行完显示

d0143d857f781cbc995385f5ffd13440.png
a7d4199308ca340c972e7a276ed91fff.png

年轻代和老年代

JVM中的对象可分为两类:一类生命周期较短,创建消亡都很快。另一类生命周期很长。分别存放到年轻代和老年代区域

默认新生代:老年代=1:2,新生代中Eden:from:to=8:1:1

调整新生代和老年代的占比(一般不去调):-XX:NewRatio=4,表示新生代占1,老年代占4,新生代占整个堆的1/5

    • 查看当前进程新生代和老年代的占比:jinfo -flag NewRatio pid
    • 设置新生代空间的大小(一般不设置,用比例就好):-Xmn

调整新生代中Eden和Survivor的占比:-XX:SurvivorRatio=8,表示Eden占8,两个Survivor各占1

    • 默认有自适应比例的机制:-XX:-UseAdaptiveSizePolicy关闭自适应分配策略
    • 查看当前比例:jinfo -flag SurvivorRatio pid

设置晋升老年代的年龄:-XX:MaxTenuringThreshold=,默认值是15

当Eden区满的时候就会触发YGC/MinorGC,将Eden和Survivor一起回收,但是Survivor区满的时候不会触发YGC!!!!!

对象分配过程

当新对象太大,Eden放不下,YGC之后依然放不下,就会直接晋升到老年代区

当YGC时,Survivor区不够放时,对象会直接晋升到老年代区

动态对象年龄判断/动态年龄计算: 如果Sruvior区中从小到大到某个年龄的所有对象大小总和大于Survivor空间的一半,则年龄大于等于该年龄的对象直接进入老年代,无需达到MaxTenuringThreshold,同时会取这个年龄和MaxTenuringThreshold的较小值作为新的晋升阈值。动态年龄计算可以避免我们手动设置的MaxTenuringThreshold过大的问题,防止survivor区溢出。此外防止MaxTenuringThreshold过小,过早晋升导致大量短期对象被晋升到老年代,老年代空间迅速增长,引起频繁的MajorGC,严重影响GC性能

c90086159e9321e065b31c2b83790c5b.png

MinorGC、MajorGC、FullGC

JVM进行GC时,并不是每次都对三个内存区域(新生代,老年代;永久代/元空间)一起回收的,大部分时候回收的都是新生代

针对HotSpot的实现,它里面把GC按照回收区域分为两类:部分收集(Partial GC),完全收集(Full GC)

部分收集:不是完整收集整个堆的垃圾

  1. 新生代收集(Minor GC/Young GC):只收集新生代的垃圾
  2. 老年代收集(Major GC/Old GC):只收集老年代的垃圾。目前只有CMS GC会单独收集老年代。多数时候MajorGC会和FullGC混淆使用,要具体分辨是老年代回收还是整个堆的回收
  3. 混合收集(Mixed GC):收集整个新生代以及部分老年代的垃圾。目前只有G1 GC会有这种行为
  4. 整堆收集:Full GC收集整个java堆和方法区的垃圾

TLAB

Thread Local Allocation Buffer(TLAB):由于堆区是线程共享的,为了避免多线程操作同一地址,通常需要使用加锁等机制,但是降低了效率。

于是JVM在Eden区中为每个线程分配了一个私有的缓存区域,使用TLAB可以避免线程安全问题,提升内存分配的吞吐量,这就是快速分配策略

TLAB区域非常小,只占整个Eden区的1%,JVM将TLAB作为内存分配的首选,可以通过-XX:TLABWasteTargetPercent设置TLAB空间占Eden的百分比大小

一旦对象在TLAB空间分配内存失败时,JVM就会尝试使用加锁机制确保数据操作的原子性,从而直接在Eden空间中分配内存

-XX:UseTLAB查看是否开启TLAB,默认是开启的

6dfa5731dbac9304ef14423d102759d1.png

堆不是对象分配的唯一选择

随着JIT编译器的发展和逃逸分析技术逐渐成熟,栈上分配、标量替换优化技术有可能导致对象分配到栈上

如果经过逃逸分析(Escape Analysis)后发现,一个对象没有逃逸出方法的话,就可能被优化成栈上分配,这样就无需在堆上分配,无需进行垃圾回收了

  • 栈上分配: JIT编译器在编译期间根据逃逸分析的结果,对于没有逃逸的对象,就可能被优化成栈上分配。(目前HotSpot只有标量替换)
  • 同步省略 / 锁消除: JIT编译器在借助逃逸分析判断同步块所使用的锁对象是否只能被一个线程访问而没有被发布到其他线程,如果是,JIT编译器在编译这个同步代码块时就取消对这部分代码的同步,提高性能,也叫锁消除
  • 分离对象 / 标量替换: 有的对象不会被外界访问到,那么对象的部分或全部可以不存储在内存(堆),而是存储到CPU寄存器(栈)中。即JIT优化把对象拆解成若干标量,存储到栈上,而不会创建对象
    • 标量:不可以再拆分成更小的数据的数据。Java中原始数据类型都是标量
    • 聚合量:可以拆分为其他聚合量和标量。如Java中的类



推荐阅读
  • 本文整理了Java面试中常见的问题及相关概念的解析,包括HashMap中为什么重写equals还要重写hashcode、map的分类和常见情况、final关键字的用法、Synchronized和lock的区别、volatile的介绍、Syncronized锁的作用、构造函数和构造函数重载的概念、方法覆盖和方法重载的区别、反射获取和设置对象私有字段的值的方法、通过反射创建对象的方式以及内部类的详解。 ... [详细]
  • Java序列化对象传给PHP的方法及原理解析
    本文介绍了Java序列化对象传给PHP的方法及原理,包括Java对象传递的方式、序列化的方式、PHP中的序列化用法介绍、Java是否能反序列化PHP的数据、Java序列化的原理以及解决Java序列化中的问题。同时还解释了序列化的概念和作用,以及代码执行序列化所需要的权限。最后指出,序列化会将对象实例的所有字段都进行序列化,使得数据能够被表示为实例的序列化数据,但只有能够解释该格式的代码才能够确定数据的内容。 ... [详细]
  • Java太阳系小游戏分析和源码详解
    本文介绍了一个基于Java的太阳系小游戏的分析和源码详解。通过对面向对象的知识的学习和实践,作者实现了太阳系各行星绕太阳转的效果。文章详细介绍了游戏的设计思路和源码结构,包括工具类、常量、图片加载、面板等。通过这个小游戏的制作,读者可以巩固和应用所学的知识,如类的继承、方法的重载与重写、多态和封装等。 ... [详细]
  • 本文总结了Java中日期格式化的常用方法,并给出了示例代码。通过使用SimpleDateFormat类和jstl fmt标签库,可以实现日期的格式化和显示。在页面中添加相应的标签库引用后,可以使用不同的日期格式化样式来显示当前年份和月份。该文提供了详细的代码示例和说明。 ... [详细]
  • 本文介绍了lua语言中闭包的特性及其在模式匹配、日期处理、编译和模块化等方面的应用。lua中的闭包是严格遵循词法定界的第一类值,函数可以作为变量自由传递,也可以作为参数传递给其他函数。这些特性使得lua语言具有极大的灵活性,为程序开发带来了便利。 ... [详细]
  • 开发笔记:加密&json&StringIO模块&BytesIO模块
    篇首语:本文由编程笔记#小编为大家整理,主要介绍了加密&json&StringIO模块&BytesIO模块相关的知识,希望对你有一定的参考价值。一、加密加密 ... [详细]
  • 本文介绍了C#中生成随机数的三种方法,并分析了其中存在的问题。首先介绍了使用Random类生成随机数的默认方法,但在高并发情况下可能会出现重复的情况。接着通过循环生成了一系列随机数,进一步突显了这个问题。文章指出,随机数生成在任何编程语言中都是必备的功能,但Random类生成的随机数并不可靠。最后,提出了需要寻找其他可靠的随机数生成方法的建议。 ... [详细]
  • 如何使用Java获取服务器硬件信息和磁盘负载率
    本文介绍了使用Java编程语言获取服务器硬件信息和磁盘负载率的方法。首先在远程服务器上搭建一个支持服务端语言的HTTP服务,并获取服务器的磁盘信息,并将结果输出。然后在本地使用JS编写一个AJAX脚本,远程请求服务端的程序,得到结果并展示给用户。其中还介绍了如何提取硬盘序列号的方法。 ... [详细]
  • 本文讨论了一个关于cuowu类的问题,作者在使用cuowu类时遇到了错误提示和使用AdjustmentListener的问题。文章提供了16个解决方案,并给出了两个可能导致错误的原因。 ... [详细]
  • Java String与StringBuffer的区别及其应用场景
    本文主要介绍了Java中String和StringBuffer的区别,String是不可变的,而StringBuffer是可变的。StringBuffer在进行字符串处理时不生成新的对象,内存使用上要优于String类。因此,在需要频繁对字符串进行修改的情况下,使用StringBuffer更加适合。同时,文章还介绍了String和StringBuffer的应用场景。 ... [详细]
  • 本文介绍了Java高并发程序设计中线程安全的概念与synchronized关键字的使用。通过一个计数器的例子,演示了多线程同时对变量进行累加操作时可能出现的问题。最终值会小于预期的原因是因为两个线程同时对变量进行写入时,其中一个线程的结果会覆盖另一个线程的结果。为了解决这个问题,可以使用synchronized关键字来保证线程安全。 ... [详细]
  • 自动轮播,反转播放的ViewPagerAdapter的使用方法和效果展示
    本文介绍了如何使用自动轮播、反转播放的ViewPagerAdapter,并展示了其效果。该ViewPagerAdapter支持无限循环、触摸暂停、切换缩放等功能。同时提供了使用GIF.gif的示例和github地址。通过LoopFragmentPagerAdapter类的getActualCount、getActualItem和getActualPagerTitle方法可以实现自定义的循环效果和标题展示。 ... [详细]
  • Java学习笔记之面向对象编程(OOP)
    本文介绍了Java学习笔记中的面向对象编程(OOP)内容,包括OOP的三大特性(封装、继承、多态)和五大原则(单一职责原则、开放封闭原则、里式替换原则、依赖倒置原则)。通过学习OOP,可以提高代码复用性、拓展性和安全性。 ... [详细]
  • 预备知识可参考我整理的博客Windows编程之线程:https:www.cnblogs.comZhuSenlinp16662075.htmlWindows编程之线程同步:https ... [详细]
  • SpringBoot整合SpringSecurity+JWT实现单点登录
    SpringBoot整合SpringSecurity+JWT实现单点登录,Go语言社区,Golang程序员人脉社 ... [详细]
author-avatar
赵乙潘_107
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有