微信公众号:Java大家族

JVM将初始和最大内存大小设置为相同值的好处

启动应用程序时,我们指定初始内存大小和最大内存大小。对于在 JVM(Java 虚拟机)上运行的应用程序,初始和最大内存大小通过 “-Xms” 和 “-Xmx” 参数指定。如果 Java 应用程序在容器上运行,则通过“-XX:InitialRAMPercentage”和“-XX:MaxRAMPercentage”参数指定它。大多数企业将初始内存大小设置为低于最大内存大小的值。与这种普遍接受的做法相反,将初始内存大小设置为与最大内存大小相同具有如下优势。让我们在这篇文章中讨论它们。

1. 可用性

假设您正在启动应用程序,初始堆大小为 2GB,最大堆大小为 24GB。这意味着当应用程序启动时,操作系统将为您的应用程序分配 2GB 的内存。从那时起,当应用程序开始处理新请求时,将分配额外的内存,直到达到最大 24GB。

假设当您的应用程序的内存消耗正在从2GB增长到24GB的过程中,此时,服务器启动了其他一些进程,并且这些进程开始消耗内存。这种情况在生产/云环境中非常常见,尤其是在应用程序与其他进程(如自定义脚本、cron 作业、监视代理等)一起运行时。

发生这种情况时,您的应用程序将遇到以下情况:

“java.lang.OutOfMemoryError:Java heap space”
操作系统将终止您的应用程序,并显示“内存不足:杀死进程。
        这意味着您的应用程序将在事务过程中崩溃。如果应用程序在启动期间以最大内存启动,则应用程序将是安全的。操作系统将仅终止内存消耗正在增长的新启动的脚本/cron 作业,而不会终止在启动期间内存已完全分配的应用程序。

2. 性能

我们还观察到,以相同的初始堆大小和最大堆大小启动的应用程序往往比以较低的初始堆大小启动的应用程序的性能相对较好。

这是一个真实的案例研究:我们使用记忆密集型应用程序进行测试。此应用程序处理非常大的二进制堆转储文件并生成分析报告。在这个应用程序中,我们反复分析一个11GB大小的二进制文件,这样它就会给操作系统带来内存压力。

我们进行了两个测试场景:

方案 1:我们将初始堆大小设置为 2GB,最大堆大小设置为 24GB。
方案 2:我们将初始堆大小和最大堆大小都设置为 24GB。
        在场景 1 中,我们观察到平均响应时间为 385.32 秒,而在场景 2 中,我们观察到平均响应时间为 366.55 秒。响应时间缩短了 5.11%。响应时间的这种改善是由于以下两个原因:

操作系统的内存分配和解除分配

GC 暂停时间影响
让我们在这里讨论它们:

从操作系统分配和解除分配内存
        当您为初始堆大小和最大堆大小设置了不同的大小时,JVM 将不得不与操作系统协商,以便在需要时分配内存。同样,当应用程序对内存的需求在运行时出现故障时,操作系统将占用分配的内存。这种持续的分配和解除分配将增加应用程序的开销。

场景 1:内存分配波动(按 GCeasy 绘制的图表)

上图显示了场景 1 JVM 的已分配和已解除分配的内存。从图表中,您可以注意到内存在不断波动(在 2GB 到 24GB 之间波动)。当应用程序处理堆转储时,内存最多可达 24GB。处理后,内存将回落到 2GB。当它再次处理新的堆转储时,内存会回弹到 24GB。

场景 2:内存分配常量(由 GCeasy 绘制)

上图显示了场景 2 JVM 在其生命周期内分配的内存。你可以看到没有波动。内存是在启动期间从操作系统保留的,从那时起,没有波动。无论应用程序中的活动如何,它始终保持在24GB。此行为有可能在一定程度上提高应用程序的性能。

GC 暂停时间影响

当垃圾回收运行时,它会暂停应用程序,这将对客户产生负面影响。我们使用 GCeasy 工具研究了两种方案的垃圾回收性能。结果如下:

垃圾回收性能结果

我们注意到 GC 吞吐量和 GC 暂停时间略有下降。在方案 1 中,GC 吞吐量为 96.59%,而在方案 2 中,GC 吞吐量略好 (97.83%)。同样,在场景 1 中,Max GC 的暂停时间为 5.23 秒,而在场景 2 中仅为 1.65 秒。。

应用程序启动时间
        如果将初始堆大小设置为与最大堆大小相同,则应用程序的启动时间也会更好。以下是 Oracle 文档的摘录:

如果初始堆太小,Java 应用程序的启动速度会变慢,因为 JVM 被迫频繁地执行垃圾回收,直到堆增长到更合理的大小。为获得最佳启动性能,请将初始堆大小设置为与最大堆大小相同。"

4. 成本

无论您将初始堆大小 (-Xms) 和最大堆大小 (-Xmx) 设置为相同值还是其他值,您支付给云托管提供商的计算成本都不会更改。假设您正在使用阿里云、腾讯云等云厂商的实例,那么无论设置初始堆大小和最大堆大小的值如何,您最终都将支付固定小时的费用。云提供商不会根据您在该计算机中使用的内存量向您收费。它们仅根据您使用实例的时间收费。因此,将初始堆大小设置为低于最大堆大小不会节省成本。

结论

在配置线程池或连接池时,将初始堆大小配置为小于最大堆大小是有意义的。在这些资源中,过度分配会产生不必要的影响,但是,内存并非如此。因此,如果要构建企业应用程序,强烈建议将初始堆大小和最大堆大小设置为相同的值。

微信公众号:Java大家族

什么?你们公司还没有将JVM初始和最大堆内存大小设置为相同值?的更多相关文章

  1. 【转】jvm 堆内存 栈内存 大小设置

    原文地址:http://blog.csdn.net/qh_java/article/details/46608395 4种方式配置不同作用域的jvm的堆栈内存! 1.Eclise 中设置jvm内存: ...

  2. jvm 堆内存 栈内存 大小设置

                        4种方式配置不同作用域的jvm的堆栈内存. 1.Eclise 中设置jvm内存: 改动eclipse的配置文件,对全部project都起作用 改动eclipse ...

  3. 你还在为了JVM而烦恼么?(内存结构和垃圾回收算法)

    ​ 做JAVA也有接近2年的时间了,公司的leader说,做JAVA,三年是个坎,如果过了三年你还没有去研究JVM的话,那么你这个程序员只能是板砖的工具了.恰逢辞职,来个JVM的解析可好? JVM是J ...

  4. jvm 虚拟机参数_堆内存分配

    1.参数 -XX:+PrintGC 只要遇到 GC 就会打印日志 -XX:+UseSerialGC 配置串行回收器 -XX:+PrintGCDetails 查看详细信息,包括各个区的情况 -XX:+P ...

  5. JVM性能调优-GC内存模型及垃圾收集算法

    JVM内存管理模型: http://developer.51cto.com/art/201002/184385.htm 一 JVM内存模型 1.1 Java栈 Java栈是与每一个线程关联的,JVM在 ...

  6. JVM heap中各generation的大小(Sizing the Generations)

    查看参数 使用 -XX:+PrintFlagsFinal 打印当前环境JVM参数默认值, 比如: java -XX:PrintFlagsFinal -version, 也可以用java [生产环境参数 ...

  7. JVM中内存的设置和分配(最大内存,总内存,剩余内存的区别)

    1.设置分配的内存大小 -vmargs -Xms128M -Xmx512M -XX:PermSize=64M -XX:MaxPermSize=128M -vmargs 说明后面是VM的参数,所以后面的 ...

  8. 深入理解JVM虚拟机11:Java内存异常原理与实践

    本文转自互联网,侵删 本系列文章将整理到我在GitHub上的<Java面试指南>仓库,更多精彩内容请到我的仓库里查看 https://github.com/h2pl/Java-Tutori ...

  9. Linux 下修改Tomcat使用的JVM内存大小

    我的服务器的配置: # OS specific support.  $var _must_ be set to either true or false. JAVA_OPTS="-Xms10 ...

随机推荐

  1. QT designer的安装与汉化(pycharm)

    QT designer的安装 1.安装好Python3的环境 添加环境变量,保证安装正确, 2.安装PyQt5 采用命令安装,Win+R,输入CMD,打开命令框,输入以下命令.后面是豆瓣的镜像地址,是 ...

  2. git tag、gitignore和git撤销提交

    前言 最近在git的使用过程中遇到了一些新的问题,所以写下来方便自己回忆. git tag 打标签 git tag -a v1.00 -m "注释" git tag 打标签命令 - ...

  3. SpringAOP 失效解决方案、Spring事务失效

    SpringAOP 失效解决方案 SpringAOP是基于代理来对目标方法进行增强,但是有的时候又会出现"增强无效"的情况,比如在@Transactional下的某类中的方法内调用 ...

  4. MySQL 面试题MySQL 中有哪几种锁?

    1.表级锁:开销小,加锁快:不会出现死锁:锁定粒度大,发生锁冲突的概率最 高,并发度最低. 2.行级锁:开销大,加锁慢:会出现死锁:锁定粒度最小,发生锁冲突的概率最 低,并发度也最高. 3.页面锁:开 ...

  5. 数组有没有 length()方法?String 有没有 length()方法?

    数组没有 length()方法,有 length 的属性. String 有 length()方法.JavaScript中,获得字符串的长度是通过 length 属性得到的,这一点容易和 Java 混 ...

  6. JAVA DAEMON线程的理解

    java线程分两种:用户线程和daemon线程.daemon线程或进程就是守护线程或者进程,但是java中所说的daemon线程和linux中的daemon是有一点区别的. linux中的daemon ...

  7. Executors 类是什么?

    Executors 为 Executor,ExecutorService,ScheduledExecutorService, ThreadFactory 和 Callable 类提供了一些工具方法. ...

  8. 1.0缓存:Login.aspx?

    所有的manifest资源配置文件以CACHE MANIFEST声明开头. #(哈希标签)有助于提供缓存文件的版本. CACHE命令指定哪些文件需要被缓存. manifest资源配置文件的内容类型应是 ...

  9. 深入 x64

      本篇原文为 X64 Deep Dive,如果有良好的英文基础的能力,可以点击该链接进行阅读.本文为我个人:寂静的羽夏(wingsummer) 中文翻译,非机翻,著作权归原作者所有.   由于原文十 ...

  10. 小小标签,强大功能——深藏不露的 <input>

    <input> 虽只是一个看似简单的 HTML 表单元素,但它这么一个单一的元素,就有多达 30 多个属性(attribute),相信无论你是个小菜鸟还是像我一样写了 15 年 HTML ...