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

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

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


结论


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

相关文章
|
27天前
|
存储 安全 Java
jvm 锁的 膨胀过程?锁内存怎么变化的
【10月更文挑战第3天】在Java虚拟机(JVM)中,`synchronized`关键字用于实现同步,确保多个线程在访问共享资源时的一致性和线程安全。JVM对`synchronized`进行了优化,以适应不同的竞争场景,这种优化主要体现在锁的膨胀过程,即从偏向锁到轻量级锁,再到重量级锁的转变。下面我们将详细介绍这一过程以及锁在内存中的变化。
36 4
|
2天前
|
弹性计算 Kubernetes Perl
k8s 设置pod 的cpu 和内存
在 Kubernetes (k8s) 中,设置 Pod 的 CPU 和内存资源限制和请求是非常重要的,因为这有助于确保集群资源的合理分配和有效利用。你可以通过定义 Pod 的 `resources` 字段来设置这些限制。 以下是一个示例 YAML 文件,展示了如何为一个 Pod 设置 CPU 和内存资源请求(requests)和限制(limits): ```yaml apiVersion: v1 kind: Pod metadata: name: example-pod spec: containers: - name: example-container image:
|
27天前
|
缓存 算法 Java
JVM知识体系学习六:JVM垃圾是什么、GC常用垃圾清除算法、堆内存逻辑分区、栈上分配、对象何时进入老年代、有关老年代新生代的两个问题、常见的垃圾回收器、CMS
这篇文章详细介绍了Java虚拟机(JVM)中的垃圾回收机制,包括垃圾的定义、垃圾回收算法、堆内存的逻辑分区、对象的内存分配和回收过程,以及不同垃圾回收器的工作原理和参数设置。
54 4
JVM知识体系学习六:JVM垃圾是什么、GC常用垃圾清除算法、堆内存逻辑分区、栈上分配、对象何时进入老年代、有关老年代新生代的两个问题、常见的垃圾回收器、CMS
|
30天前
|
存储 缓存 算法
JVM核心知识点整理(内存模型),收藏再看!
JVM核心知识点整理(内存模型),收藏再看!
JVM核心知识点整理(内存模型),收藏再看!
|
17天前
|
存储 算法 Java
聊聊jvm的内存结构, 以及各种结构的作用
【10月更文挑战第27天】JVM(Java虚拟机)的内存结构主要包括程序计数器、Java虚拟机栈、本地方法栈、Java堆、方法区和运行时常量池。各部分协同工作,为Java程序提供高效稳定的内存管理和运行环境,确保程序的正常执行、数据存储和资源利用。
44 10
|
16天前
|
存储 算法 Java
Java虚拟机(JVM)的内存管理与性能优化
本文深入探讨了Java虚拟机(JVM)的内存管理机制,包括堆、栈、方法区等关键区域的功能与作用。通过分析垃圾回收算法和调优策略,旨在帮助开发者理解如何有效提升Java应用的性能。文章采用通俗易懂的语言,结合具体实例,使读者能够轻松掌握复杂的内存管理概念,并应用于实际开发中。
|
26天前
|
存储 监控 算法
JVM调优深度剖析:内存模型、垃圾收集、工具与实战
【10月更文挑战第9天】在Java开发领域,Java虚拟机(JVM)的性能调优是构建高性能、高并发系统不可或缺的一部分。作为一名资深架构师,深入理解JVM的内存模型、垃圾收集机制、调优工具及其实现原理,对于提升系统的整体性能和稳定性至关重要。本文将深入探讨这些内容,并提供针对单机几十万并发系统的JVM调优策略和Java代码示例。
46 2
|
27天前
|
存储 Java
JVM知识体系学习四:排序规范(happens-before原则)、对象创建过程、对象的内存中存储布局、对象的大小、对象头内容、对象如何定位、对象如何分配
这篇文章详细地介绍了Java对象的创建过程、内存布局、对象头的MarkWord、对象的定位方式以及对象的分配策略,并深入探讨了happens-before原则以确保多线程环境下的正确同步。
50 0
JVM知识体系学习四:排序规范(happens-before原则)、对象创建过程、对象的内存中存储布局、对象的大小、对象头内容、对象如何定位、对象如何分配
|
30天前
|
Java API 对象存储
JVM进阶调优系列(2)字节面试:JVM内存区域怎么划分,分别有什么用?
本文详细解析了JVM类加载过程的关键步骤,包括加载验证、准备、解析和初始化等阶段,并介绍了元数据区、程序计数器、虚拟机栈、堆内存及本地方法栈的作用。通过本文,读者可以深入了解JVM的工作原理,理解类加载器的类型及其机制,并掌握类加载过程中各阶段的具体操作。
|
1月前
|
存储 Java Linux
【JVM】JVM执行流程和内存区域划分
【JVM】JVM执行流程和内存区域划分
41 1