首页 > 其他分享 >JVM调优总结 -Xms -Xmx -Xmn -Xss

JVM调优总结 -Xms -Xmx -Xmn -Xss

时间:2022-12-27 20:32:29浏览次数:90  
标签:Xss Xmn java 收集器 XX 调优 GC 内存 JVM

 

java启动参数共分为三类;
其一是标准参数(-),所有的JVM实现都必须实现这些参数的功能,而且向后兼容;
其二是非标准参数(-X),默认jvm实现这些参数的功能,但是并不保证所有jvm实现都满足,且不保证向后兼容;
其三是非Stable参数(-XX),此类参数各个jvm实现会有所不同,将来可能会随时取消,需要慎重使用;

标准参数中比较有用的:

verbose
-verbose:class
输出jvm载入类的相关信息,当jvm报告说找不到类或者类冲突时可此进行诊断。
-verbose:gc
输出每次GC的相关情况。
-verbose:jni
输出native方法调用的相关情况,一般用于诊断jni调用错误信息。

非标准参数又称为扩展参数

一般用到最多的是
-Xms512m 设置JVM促使内存为512m。此值可以设置与-Xmx相同,以避免每次垃圾回收完成后JVM重新分配内存。
-Xmx512m ,设置JVM最大可用内存为512M。
-Xmn200m:设置年轻代大小为200M。整个堆大小=年轻代大小 + 年老代大小 + 持久代大小。持久代一般固定大小为64m,所以增大年轻代后,将会减小年老代大小。此值对系统性能影响较大,Sun官方推荐配置为整个堆的3/8。
-Xss128k:设置每个线程的堆栈大小。JDK5.0以后每个线程堆栈大小为1M,以前每个线程堆栈大小为256K。更具应用的线程所需内存大小进行调整。在相同物理内 存下,减小这个值能生成更多的线程。但是操作系统对一个进程内的线程数还是有限制的,不能无限生成,经验值在3000~5000左右。

-Xloggc:file
与-verbose:gc功能类似,只是将每次GC事件的相关情况记录到一个文件中,文件的位置最好在本地,以避免网络的潜在问题。
若与verbose命令同时出现在命令行中,则以-Xloggc为准。
-Xprof

跟踪正运行的程序,并将跟踪数据在标准输出输出;适合于开发环境调试。

用-XX作为前缀的参数列表在jvm中可能是不健壮的,SUN也不推荐使用,后续可能会在没有通知的情况下就直接取消了;但是由于这些参数中的确有很多是对我们很有用的,比如我们经常会见到的-XX:PermSize、-XX:MaxPermSize等等;


方法区:

方法区也是所有线程共享。主要用于存储类的信息、常量池、方法数据、方法代码等。方法区逻辑上属于堆的一部分,但是为了与堆进行区分,通常又叫“非堆”。 关于方法区内存溢出的问题会在下文中详细探讨。

方法区和“PermGen space”又有着本质的区别。前者是 JVM 的规范,而后者则是 JVM 规范的一种实现,并且只有 HotSpot 才有 “PermGen space”,而对于其他类型的虚拟机,如 JRockit(Oracle)、J9(IBM) 并没有“PermGen space”。

元空间的本质和永久代类似,都是对JVM规范中方法区的实现。不过元空间与永久代之间最大的区别在于:元空间并不在虚拟机中,而是使用本地内存。因此,默认情况下,元空间的大小仅受本地内存限制;

二、PermGen(永久代)

  绝大部分 Java 程序员应该都见过 "java.lang.OutOfMemoryError: PermGen space "这个异常。这里的 “PermGen space”其实指的就是方法区。
不过方法区和“PermGen space”又有着本质的区别。前者是 JVM 的规范,而后者则是 JVM 规范的一种实现,并且只有 HotSpot 才有 “PermGen space”,

JVM 种类有很多,比如 Oralce-Sun Hotspot, Oralce JRockit, IBM J9, Taobao JVM(淘宝好样的!)等等。当然武林盟主是Hotspot了,这个毫无争议。需要注意的是,PermGen space是Oracle-Sun Hotspot才有,JRockit以及J9是没有这个区域。

由于方法区主要存储类的相关信息,所以对于动态生成类的情况比较容易出现永久代的内存溢出。最典型的场景就是,在 jsp 页面比较多的情况,容易出现永久代内存溢出。

JDK 1.7,是因为在 JDK 1.8 中, HotSpot 已经没有 “PermGen space”这个区间了,取而代之是一个叫做 Metaspace(元空间) 的东西。下面我们就来看看 Metaspace 与 PermGen space 的区别。

三、Metaspace(元空间)

  其实,移除永久代的工作从JDK1.7就开始了。JDK1.7中,存储在永久代的部分数据就已经转移到了Java Heap或者是 Native Heap。但永久代仍存在于JDK1.7中,并没完全移除,譬如符号引用(Symbols)转移到了native heap;字面量(interned strings)转移到了java heap;类的静态变量(class statics)转移到了java heap。

JVM调优总结 -Xms -Xmx -Xmn -Xss_XSS

 

JVM调优总结 -Xms -Xmx -Xmn -Xss_jvm_02

 

 

JVM调优总结 -Xms -Xmx -Xmn -Xss_XSS_03

 

 

  从上述结果可以看出,JDK 1.6下,会出现“PermGen Space”的内存溢出,而在 JDK 1.7和 JDK 1.8 中,会出现堆内存溢出,并且 JDK 1.8中 PermSize 和 MaxPermGen 已经无效。因此,可以大致验证 JDK 1.7 和 1.8 将字符串常量由永久代转移到堆中,并且 JDK 1.8 中已经不存在永久代的结论。现在我们看看元空间到底是一个什么东西?

  元空间的本质和永久代类似,都是对JVM规范中方法区的实现。不过元空间与永久代之间最大的区别在于:元空间并不在虚拟机中,而是使用本地内存。因此,默认情况下,元空间的大小仅受本地内存限制,但可以通过以下参数来指定元空间的大小:

  -XX:MetaspaceSize,初始空间大小,达到该值就会触发垃圾收集进行类型卸载,同时GC会对该值进行调整:如果释放了大量的空间,就适当降低该值;如果释放了很少的空间,那么在不超过MaxMetaspaceSize时,适当提高该值。
  -XX:MaxMetaspaceSize,最大空间,默认是没有限制的。

  除了上面两个指定大小的选项以外,还有两个与 GC 相关的属性:
  -XX:MinMetaspaceFreeRatio,在GC之后,最小的Metaspace剩余空间容量的百分比,减少为分配空间所导致的垃圾收集
  -XX:MaxMetaspaceFreeRatio,在GC之后,最大的Metaspace剩余空间容量的百分比,减少为释放空间所导致的垃圾收集

现在我们在 JDK 8下重新运行一下代码段 4,不过这次不再指定 PermSize 和 MaxPermSize。而是指定 MetaSpaceSize 和 MaxMetaSpaceSize的大小。输出结果如下:

JVM调优总结 -Xms -Xmx -Xmn -Xss_Xms_04

 

四、总结
  通过上面分析,大家应该大致了解了 JVM 的内存划分,也清楚了 JDK 8 中永久代向元空间的转换。不过大家应该都有一个疑问,就是为什么要做这个转换?所以,最后给大家总结以下几点原因:
  1、字符串存在永久代中,容易出现性能问题和内存溢出。
  2、类及方法的信息等比较难确定其大小,因此对于永久代的大小指定比较困难,太小容易出现永久代溢出,太大则容易导致老年代溢出。
  3、永久代会为 GC 带来不必要的复杂度,并且回收效率偏低。
  4、Oracle 可能会将HotSpot 与 JRockit 合二为一。

java -XX:+PrintFlagsFinal -version |grep MetaspaceSize

 

为什么移除持久代
它的大小是在启动时固定好的——很难进行调优。-XX:MaxPermSize,设置成多少好呢?
HotSpot的内部类型也是Java对象:它可能会在Full GC中被移动,同时它对应用不透明,且是非强类型的,难以跟踪调试,还需要存储元数据的元数据信息(meta-metadata)。
简化Full GC:每一个回收器有专门的元数据迭代器。
可以在GC不进行暂停的情况下并发地释放类数据。
使得原来受限于持久代的一些改进未来有可能实现
根据上面的各种原因,永久代最终被移除,方法区移至Metaspace,字符串常量移至Java Heap。

1.3、移除持久代后,PermGen空间的状况
这部分内存空间将全部移除。

JVM的参数:PermSize 和 MaxPermSize 会被忽略并给出警告(如果在启用时设置了这两个参数)。

JVM调优总结 -Xms -Xmx -Xmn -Xss_Xmn_05

 

 

JVM调优总结 -Xms -Xmx -Xmn -Xss_jvm_06

 

 

二、元空间
随着JDK8的到来,JVM不再有PermGen。但类的元数据信息(metadata)还在,只不过不再是存储在连续的堆空间上,而是移动到叫做“Metaspace”的本地内存(Native memory)中。

JVM调优总结 -Xms -Xmx -Xmn -Xss_Xms_07

2.2、元空间的特点
充分利用了Java语言规范中的好处:类及相关的元数据的生命周期与类加载器的一致。
每个加载器有专门的存储空间
只进行线性分配
不会单独回收某个类
省掉了GC扫描及压缩的时间
元空间里的对象的位置是固定的
如果GC发现某个类加载器不再存活了,会把相关的空间整个回收掉

 

五、Metaspace可以使用的工具
针对Metaspace,JDK自带的一些工具做了修改来展示Metaspace的信息:

jmap -clstats :打印类加载器的统计信息(取代了在JDK8之前打印类加载器信息的permstat)。
jstat -gc :Metaspace的信息也会被打印出来。
jcmd GC.class_stats:这是一个新的诊断命令,可以使用户连接到存活的JVM,转储Java类元数据的详细统计。
示例1:jmap -clstats

[ciadmin@2-103test_app pos-gateway-cloud]$ jmap -clstats 26964
Attaching to process ID 26964, please wait...
Debugger attached successfully.
Server compiler detected.
JVM version is 25.131-b11
finding class loader instances ..done.
computing per loader stat ..done.
please wait.. computing liveness..................................................................................liveness analysis may be inaccurate ...
class_loader classes bytes parent_loader alive? type

<bootstrap> 2699 4611703 null live <internal>
0x00000000a1013a00 1 880 0x00000000a001b938 dead sun/reflect/DelegatingClassLoader@0x0000000100009df8
0x00000000a3e931e8 1 880 null dead sun/reflect/DelegatingClassLoader@0x0000000100009df8
0x00000000a083d280 1 1471 0x00000000a001b938 dead sun/reflect/DelegatingClassLoader@0x0000000100009df8
0x00000000a1c057c8 1 880 0x00000000a001b938 dead sun/reflect/DelegatingClassLoader@0x0000000100009df8
0x00000000a1013938 1 1474 0x00000000a001b938 dead sun/reflect/DelegatingClassLoader@0x0000000100009df8
0x00000000a1013d38 1 1471 0x00000000a001b938 dead sun/reflect/DelegatingClassLoader@0x0000000100009df8
0x00000000a141ae78 1 880 0x00000000a001b938 dead sun/reflect/DelegatingClassLoader@0x0000000100009df8
0x00000000a083d1b8 1 1473 0x00000000a001b938 dead sun/reflect/DelegatingClassLoader@0x0000000100009df8
0x00000000a163c658 1 880 0x00000000a001b938 dead sun/reflect/DelegatingClassLoader@0x0000000100009df8
0x00000000a293afa8 1 1473 0x00000000a001b938 dead sun/reflect/DelegatingClassLoader@0x0000000100009df8
0x00000000a19ec0a0 15 70893 0x00000000a001b938 live com/aliyun/openservices/shade/com/alibaba/fastjson/util/ASMClassLoader@0x000000010066b7a0
0x00000000a2778848 1 1474 0x00000000a001b938 dead sun/reflect/DelegatingClassLoader@0x0000000100009df8
0x00000000a141a900 1 880 0x00000000a001b938 dead sun/reflect/DelegatingClassLoader@0x0000000100009df8
0x00000000a083d8c0 1 1473 0x00000000a001b938 dead sun/reflect/DelegatingClassLoader@0x0000000100009df8
0x00000000a163c720 1 880 0x00000000a001b938 dead sun/reflect/DelegatingClassLoader@0x0000000100009df8
...
0x00000000a094fe68 0 0 0x00000000a0007438 live java/net/URLClassLoader@0x000000010000ecd0

total = 177 12836 20539140 N/A alive=9, dead=168 N/A
[ciadmin@2-103test_app pos-gateway-cloud]$

示例二:jstat -gc 26964

[ciadmin@2-103test_app pos-gateway-cloud]$ jstat -gc 26964
S0C S1C S0U S1U EC EU OC OU MC MU CCSC CCSU YGC YGCT FGC FGCT GCT
3072.0 3072.0 2384.8 0.0 62976.0 6699.1 445440.0 67911.5 69760.0 68124.8 8320.0 7929.0 3792 36.649 12 1.971 38.620
[ciadmin@2-103test_app pos-gateway-cloud]$

MC:Current Metaspace Capacity(KB);
MU::Metaspace Utilization(KB)

示例三:jcmd 5943 GC.class_stats

[ciadmin@2-103test_app pos-gateway-cloud]$ jcmd 5943 GC.class_stats
5943:
GC.class_stats command requires -XX:+UnlockDiagnosticVMOptions
[ciadmin@2-103test_app pos-gateway-cloud]$

说是:应用程序启动时增加-XX:+UnlockDiagnosticVMOptions参数
加了上面的参数后,重新来一把如下:

D:\jvm>jcmd 4332 GC.class_stats
4332:
Index Super InstBytes KlassBytes annotations CpAll MethodCount Bytecodes MethodAll ROAll RWAll Total ClassName
1 -1 258458040 480 0 0 0 0 0 24 584 608 [Ljava.lang.Object;
2 368 217343856 1000 0 6864 51 3955 13744 8664 13888 22552 java.util.HashMap
3 368 144895744 1432 0 15584 93 9536 37136 19104 37048 56152 java.util.concurrent.ConcurrentHashMap
4 363 99615560 928 0 8232 24 1719 6040 4392 11304 15696 java.net.URLClassLoader
5 -1 90560256 480 0 0 0 0 0 32 584 616 [Ljava.util.concurrent.ConcurrentHashMap$Node;
6 -1 90559840 480 0 0 0 0 0 32 584 616 [Ljava.util.WeakHashMap$Entry;
7 367 72447872 1384 0 5288 59 2245 13544 7080 14016 21096 java.util.Vector
8 367 54335928 1320 0 4936 49 2359 12104 6600 12592 19192 java.util.ArrayList
9 368 54335904 976 0 4952 32 1845 11968 4856 13744 18600 java.util.WeakHashMap
10 14 54335856 656 0 7488 33 1513 8504 4792 12496 17288 sun.misc.URLClassPath
11 14 45280240 504 0 4960 27 2551 11792 5232 12536 17768 java.security.AccessControlContext
12 14 45279920 528 0 4328 12 1024 3760 2488 6496 8984 java.security.ProtectionDomain
13 14 36226208 568 0 1344 8 223 1744 1024 2952 3976 java.util.concurrent.ConcurrentHashMap$Node
14 -1 36224752 496 0 1144 14 109 2520 1112 3272 4384 java.lang.Object
15 14 36224032 552 0 1840 7 410 2744 1288 4160 5448 java.lang.ref.ReferenceQueue
16 14 36223936 552 0 5320 14 1796 4648 3552 7328 10880 java.security.CodeSource
17 7 36223936 1424 0 864 6 88 1664 704 3552 4256 java.util.Stack
18 373 27167952 1008 0 808 4 69 1000 592 2528 3120 java.util.Collections$SynchronizedSet
19 -1 27167880 480 0 0 0 0 0 24 584 608 [Ljava.security.ProtectionDomain;
20 14 18112048 496 0 360 2 10 920 216 1720 1936 java.lang.ref.ReferenceQueue$Lock
21 -1 18111968 480 0 0 0 0 0 24 584 608 [Ljava.security.Principal;
...
484 14 0 496 0 1416 20 737 3736 2240 3680 5920 sun.util.locale.LocaleUtils
1535868936 298000 1536 955600 6934 264621 1445736 885528 1980368 2865896 Total
53591.2% 10.4% 0.1% 33.3% - 9.2% 50.4% 30.9% 69.1% 100.0%
Index Super InstBytes KlassBytes annotations CpAll MethodCount Bytecodes MethodAll ROAll RWAll Total ClassName

D:\jvm>


元空间与直接内存的关系

Metaspace内存管理

在metaspace中,类和其元数据的生命周期与其对应的类加载器相同,只要类的类加载器是存活的,在Metaspace中的类元数据也是存活的,不能被回收。
每个加载器有单独的存储空间。
省掉了GC扫描及压缩的时间。
当GC发现某个类加载器不再存活了,会把对应的空间整个回收。

直接内存:直接内存主要被 Java NIO 使用,某种程度上也就是指DirectByteBuffer对象占用的堆外内存。

DirectByteBuffer对象创建时会通过Unsafe类接口直接调用操作系统的malloc分配内存,然后将内存的起始地址和大小保存下来,据此就可以直接操作内存空间

JVM调优总结 -Xms -Xmx -Xmn -Xss_Xmx_08

 

 

可以看出,直接内存的大小并不受到java堆大小的限制,甚至不受到JVM进程内存大小的限制。它只受限于本机总内存(RAM及SWAP区或者分页文件)大小以及处理器寻址空间的限制(最常见的就是32位/64位CPU的最大寻址空间限制不同)。

DirectByteBuffer使用直接内存的原因有两点:

1) 这块内存真正的分配并不在 Java 堆中,堆中只有一个很小的对象引用,这种方式能减轻GC压力
2)对于堆内对象,进行IO操作(Socket、文件读写)时需要先把对象复制一份到堆外内存再写入 Socket 或者文件,而当 DirectByteBuffer 就在堆外分配内存时可以省掉一次从堆内拷贝到堆外的操作,性能表现会更好

直接内存的回收

需注意堆外内存并不直接控制于JVM,这些内存只有在DirectByteBuffer回收掉之后才有机会被回收,而 Young GC 的时候只会将年轻代里不可达的DirectByteBuffer对象及其直接内存回收,如果这些对象大部分都晋升到了年老代,那么只能等到Full GC的时候才能彻底地回收DirectByteBuffer对象及其关联的堆外内存。因此,堆外内存的回收依赖于 Full GC

Full GC一般发生在年老代垃圾回收或者代码调用System.gc的时候,依靠年老代垃圾回收触发 Full GC,进而实现堆外内存的回收显然具有太大的不确定性。如果年老代一直不进行垃圾回收,那么堆外内存就得不到回收,机器的物理内存可能就会被慢慢耗光。为了避免这种情况发生,可以通过参数-XX:MaxDirectMemorySize来指定最大的直接内存大小,当其使用达到了阈值的时候将调用System.gc来做一次Full GC,从而完成可控的堆外内存回收。这样做的问题在于,堆外内存的回收依赖于代码调用 System.gc,先捕获到异常,再在Catch块里面通过System.gc()命 令来触发垃圾收集。但如果Java虚拟机再打开了-XX:+DisableExplicitGC开关,禁止了人工触发垃圾 收集的话,根本不会触发Full GC,这样在使用Netty等 NIO 框架时需注意是否会因为这个参数导致直接内存的泄露

直接内存出现OutOfMemoryError的原因是对该区域进行内存分配时,其内存与其他内存加起来超过最大物理内存限制(包括物理的和操作系统级的限制),从而导致OutOfMemoryError。
  另外,若我们通过参数“-XX:MaxDirectMemorySize”指定了直接内存的最大值,其超过指定的最大值时,也会抛出内存溢出异常。

-XX:MaxDirectMemorySize 参数没有指定的话,那么根据directMemory = Runtime.getRuntime().maxMemory()最大直接内存的值和堆内存大小差不多


六、提高GC的性能
如果你理解了元空间的概念,很容易发现GC的性能得到了提升。

Full GC中,元数据指向元数据的那些指针都不用再扫描了。很多复杂的元数据扫描的代码(尤其是CMS里面的那些)都删除了。
元空间只有少量的指针指向Java堆。这包括:类的元数据中指向java/lang/Class实例的指针;数组类的元数据中,指向java/lang/Class集合的指针。
没有元数据压缩的开销
减少了根对象的扫描(不再扫描虚拟机里面的已加载类的字典以及其它的内部哈希表)
减少了Full GC的时间
G1回收器中,并发标记阶段完成后可以进行类的卸载
java8中metaspace总结如下:

PermGen 空间的状况
这部分内存空间将全部移除。

JVM的参数:PermSize 和 MaxPermSize 会被忽略并给出警告(如果在启用时设置了这两个参数)。

Metaspace 内存分配模型
大部分类元数据都在本地内存中分配。

用于描述类元数据的“klasses”已经被移除。

 

Metaspace包含那几部分及存放什么数据:

一些看过Java8以后的垃圾回收日志的同学一般会对这么一句话感到很困惑:

Metaspace used 2425K, capacity 4498K, committed 4864K, reserved 1056768K
class space used 262K, capacity 386K, committed 512K, reserved 1048576K

第一个问题是,按照一般的理解,metaspace似乎是一个整体,怎么还分成了四个部分?

第二个问题是,怎么metaspace之后,还有一个class space?两者之间的关系是怎样的?

先来看第一个问题。
在Oracle文档(https://docs.oracle.com/javase/8/docs/technotes/guides/vm/gctuning/considerations.html)里面的最后一段有对这个问题的简单解释:

In the line beginning with Metaspace, the used value is the amount of space used for loaded classes. The capacity value is the space available for metadata in currently allocated chunks. The committed value is the amount of space available for chunks. The reserved value is the amount of space reserved (but not necessarily committed) for metadata. The line beginning with class space line contains the corresponding values for the metadata for compressed class pointers.

JVM调优总结 -Xms -Xmx -Xmn -Xss_jvm_09

 

 

首先可以看到的是,这些​​used​​​,​​capacity​​​,​​committed​​​和​​reserved​​并不纯粹是JVM的概念,它和操作系统相关。

先来看​​committed​​​和​​reserved​​​。​​reserved​​是指,操作系统已经为该进程“保留”的。所谓的保留,更加接近一种记账的概念,就是操作系统承诺说一大块连续的内存已经是你这个进程的了。注意的是,这里强调的是连续的内存,并且强调的是一种名义归属。那么实际上这一大块内存有没有真实对应的物理内存呢?答案是不知道。

那么什么时候才知道呢?等进程​​committed​​的时候。当进程真的要用这个连续地址空间的时候,操作系统才会分配真正的内存。所以,这也就是意味着,这个过程会失败。

举个例子来说,就好比杨白劳找黄世仁借粮,杨白劳说借我一百斤,黄世仁在本子上记录了一笔说可以,借给你了,但是现在不能给你,等你要吃的时候再来领。于是杨白劳过了一个星期,领了三十斤,这三十斤就是​​committed​​​的,之前那一百斤就是​​reserved​​的。很显然,如果黄世仁给太多人都记了一笔,说不定等杨白劳去借的时候,就没有了。

因此我才说,这个​​reserved​​更加接近记账的概念。

​used​​​和​​capacity​​​就是JVM的概念了。这两个概念非常接近JVM一些集合框架的概念。一些Java集合框架,比如某种List的实现,会有​​size​​​和​​capacity​​​的概念。比如说​​ArrayList​​​的实现里面就有​​capacity​​​和​​size​​​的概念。假如说我创建了一个可以存放20个元素的​​ArrayList​​​,但是我实际上只放了10个元素,那么​​capacity​​​就是20,而​​size​​​就是10.这里的​​size​​​和​​used​​就是一个概念。那么“元素”则是一个个内存块"block“。

​capacity​​​和​​committed​​​的关系也可以此类比,只不过​​capacity​​​反而对应到​​used​​​,​​committed​​​对应到​​capacity​​​,而所谓的”元素“,就是​​chunk​​。

至于​​class space​​​,要记住的是,​​metaspace​​​并不是全部用来放类对象的。比如说,因为每一个​​ClassLoader​​​都被分配了一块内存,这块内存可能并没有被用完,于是就会有一些内存碎片;​​metaspace​​​还需要放所谓静态变量。所以,​​class space​​​是指实际上被用于放​​class​​​的那块内存的和。
​​​https://www.jianshu.com/p/cd34d6f3b5b4​

JVM内存类别
最重要的JVM内存类别有:
Heap 堆-堆是存储类实例化或“对象”的地方。
Thread stacks 线程堆栈-每个线程都有自己的调用堆栈。堆栈存储原始局部变量和对象引用以及调用堆栈(方法调用列表)本身。当堆栈帧移出上下文时,堆栈将被清理,因此此处不执行GC。
Metaspace 元空间-Metaspace存储对象的类定义和其他一些元数据。
Code cache 代码缓存-JIT编译器将它生成的本机代码存储在代码缓存中,通过重用来提高性能。
Buffer pools 缓冲池-许多库和框架在堆外分配缓冲区以提高性能。这些缓冲池可以用来在Java代码和本机代码之间共享内存,或者将文件的区域映射到内存中。
操作系统内存——操作系统保持Java进程的堆和堆栈独立于JVM本身管理的堆和堆栈。加载的每个本机库也会消耗内存(例如libjvm.so文件). 这个通常很小

观察元空间Metaspace
Metaspace包含有关JVM正在运行的应用程序的元数据。它包含类定义、方法定义和有关程序的其他信息。加载到应用程序中的类越多,元空间就越大。

在旧版本的Java中,类元数据存储在堆中,这意味着对于一般开发人员来说,它并不是那么不可见。但是随着java8中Metaspace的引入,我们必须小心地显式地观察它。

大多数Java应用程序运行时的元空间都不到100mb,但是JRuby和Scala等其他JVM语言通常最多只能使用200mb。这是因为这些语言本质上是非常大的框架。他们正在Java标准库的顶部加载一个完整的标准库。
​​​http://javakk.com/831.html​


  1. 堆大小设置JVM 中最大堆大小有三方面限制:
    (1)相关操作系统的数据模型(32-bt还是64-bit)限制;32位系统下,一般限制在1.5G~2G;64为操作系统对内存无限制。我在Windows Server 2003 系统,3.5G物理内存,JDK5.0下测试,最大可设置为1478m。
    (2)系统的可用虚拟内存限制;
    (3)系统的可用物理内存限制。

    典型设置:
  • java -Xmx3550m -Xms3550m -Xmn2g -Xss128k
    -Xmx3550m:设置JVM最大可用内存为3550M。
    -Xms3550m:设置JVM促使内存为3550m。此值可以设置与-Xmx相同,以避免每次垃圾回收完成后JVM重新分配内存。
    -Xmn2g:设置年轻代大小为2G。整个JVM内存大小=年轻代大小 + 年老代大小 + 持久代大小。持久代一般固定大小为64m,所以增大年轻代后,将会减小年老代大小。此值对系统性能影响较大,Sun官方推荐配置为整个堆的3/8。
    -Xss256k:设置每个线程的堆栈大小。JDK5.0以后每个线程堆栈大小为1M,以前每个线程堆栈大小为256K。更具应用的线程所需内存大小进行调整。在相同物理内存下,减小这个值能生成更多的线程。但是操作系统对一个进程内的线程数还是有限制的,不能无限生成,经验值在3000~5000左右。
  • java -Xmx3550m -Xms3550m -Xss128k -XX:NewRatio=4 -XX:SurvivorRatio=4 -XX:MaxPermSize=16m -XX:MaxTenuringThreshold=0
    -XX:NewRatio=4:设置年轻代(包括Eden和两个Survivor区)与年老代的比值(除去持久代)。设置为4,则年轻代与年老代所占比值为1:4,年轻代占整个堆栈的1/5
    -XX:SurvivorRatio=4:设置年轻代中Eden区与Survivor区的大小比值。设置为4,则两个Survivor区与一个Eden区的比值为2:4,一个Survivor区占整个年轻代的1/6
    -XX:MaxPermSize=16m:设置持久代大小为16m。
    -XX:MaxTenuringThreshold=0:设置垃圾最大年龄。如果设置为0的话,则年轻代对象不经过Survivor区,直接进入年老代。对于年老代比较多的应用,可以提高效率。如果将此值设置为一个较大值,则年轻代对象会在Survivor区进行多次复制,这样可以增加对象再年轻代的存活时间,增加在年轻代即被回收的概论。
  1. 回收器选择JVM给了三种选择:串行收集器、并行收集器、并发收集器,但是串行收集器只适用于小数据量的情况,所以这里的选择主要针对并行收集器和并发收集器。默认情况下,JDK5.0以前都是使用串行收集器,如果想使用其他收集器需要在启动时加入相应参数。JDK5.0以后,JVM会根据当前​​系统配置​​进行判断。
  1. 吞吐量优先的并行收集器
    如上文所述,并行收集器主要以到达一定的吞吐量为目标,适用于科学技术和后台处理等。
    典型配置
  • java -Xmx3800m -Xms3800m -Xmn2g -Xss128k -XX:+UseParallelGC -XX:ParallelGCThreads=20
    -XX:+UseParallelGC:选择垃圾收集器为并行收集器。此配置仅对年轻代有效。即上述配置下,年轻代使用并发收集,而年老代仍旧使用串行收集。-XX:ParallelGCThreads=20:配置并行收集器的线程数,即:同时多少个线程一起进行垃圾回收。此值最好配置与处理器数目相等。
  • java -Xmx3550m -Xms3550m -Xmn2g -Xss128k -XX:+UseParallelGC -XX:ParallelGCThreads=20 -XX:+UseParallelOldGC-XX:+UseParallelOldGC:配置年老代垃圾收集方式为并行收集。JDK6.0支持对年老代并行收集。
  • java -Xmx3550m -Xms3550m -Xmn2g -Xss128k -XX:+UseParallelGC  -XX:MaxGCPauseMillis=100-XX:MaxGCPauseMillis=100:设置每次年轻代垃圾回收的最长时间,如果无法满足此时间,JVM会自动调整年轻代大小,以满足此值。
  • java -Xmx3550m -Xms3550m -Xmn2g -Xss128k -XX:+UseParallelGC  -XX:MaxGCPauseMillis=100 -XX:+UseAdaptiveSizePolicy
    -XX:+UseAdaptiveSizePolicy
    :设置此选项后,并行收集器会自动选择年轻代区大小和相应的Survivor区比例,以达到目标系统规定的最低相应时间或者收集频率等,此值建议使用并行收集器时,一直打开。
  1. 响应时间优先的并发收集器
    如上文所述,并发收集器主要是保证系统的响应时间,减少垃圾收集时的停顿时间。适用于应用服务器、电信领域等。
    典型配置
  • java -Xmx3550m -Xms3550m -Xmn2g -Xss128k -XX:ParallelGCThreads=20 -XX:+UseConcMarkSweepGC -XX:+UseParNewGC-XX:+UseConcMarkSweepGC:设置年老代为并发收集。测试中配置这个以后,-XX:NewRatio=4的配置失效了,原因不明。所以,此时年轻代大小最好用-Xmn设置。
    -XX:+UseParNewGC:设置年轻代为并行收集。可与CMS收集同时使用。JDK5.0以上,JVM会根据系统配置自行设置,所以无需再设置此值。
  • java -Xmx3550m -Xms3550m -Xmn2g -Xss128k -XX:+UseConcMarkSweepGC -XX:CMSFullGCsBeforeCompaction=5 -XX:+UseCMSCompactAtFullCollection
    -XX:CMSFullGCsBeforeCompaction:由于并发收集器不对内存空间进行压缩、整理,所以运行一段时间以后会产生“碎片”,使得运行效率降低。此值设置运行多少次GC以后对内存空间进行压缩、整理。
    -XX:+UseCMSCompactAtFullCollection:打开对年老代的压缩。可能会影响性能,但是可以消除碎片
  1. 辅助信息JVM提供了大量命令行参数,打印信息,供调试使用。主要有以下一些:
  • -XX:+PrintGC输出形式:[GC 118250K->113543K(130112K), 0.0094143 secs]                [Full GC 121376K->10414K(130112K), 0.0650971 secs]
  • -XX:+PrintGCDetails输出形式:[GC [DefNew: 8614K->781K(9088K), 0.0123035 secs] 118250K->113543K(130112K), 0.0124633 secs]                [GC [DefNew: 8614K->8614K(9088K), 0.0000665 secs][Tenured: 112761K->10414K(121024K), 0.0433488 secs] 121376K->10414K(130112K), 0.0436268 secs]
  • -XX:+PrintGCTimeStamps -XX:+PrintGC:PrintGCTimeStamps可与上面两个混合使用
    输出形式:11.851: [GC 98328K->93620K(130112K), 0.0082960 secs]
  • -XX:+PrintGCApplicationConcurrentTime:打印每次垃圾回收前,程序未中断的执行时间。可与上面混合使用
    输出形式:Application time: 0.5291524 seconds
  • -XX:+PrintGCApplicationStoppedTime:打印垃圾回收期间程序暂停的时间。可与上面混合使用
    输出形式:Total time for which application threads were stopped: 0.0468229 seconds
  • -XX:PrintHeapAtGC:打印GC前后的详细堆栈信息
    输出形式:
    34.702: [GC {Heap before gc invocations=7:
     def new generation   total 55296K, used 52568K [0x1ebd0000, 0x227d0000, 0x227d0000)
    eden space 49152K,  99% used [0x1ebd0000, 0x21bce430, 0x21bd0000)
    from space 6144K,  55% used [0x221d0000, 0x22527e10, 0x227d0000)
      to   space 6144K,   0% used [0x21bd0000, 0x21bd0000, 0x221d0000)
     tenured generation   total 69632K, used 2696K [0x227d0000, 0x26bd0000, 0x26bd0000)
    the space 69632K,   3% used [0x227d0000, 0x22a720f8, 0x22a72200, 0x26bd0000)
     compacting perm gen  total 8192K, used 2898K [0x26bd0000, 0x273d0000, 0x2abd0000)
       the space 8192K,  35% used [0x26bd0000, 0x26ea4ba8, 0x26ea4c00, 0x273d0000)
        ro space 8192K,  66% used [0x2abd0000, 0x2b12bcc0, 0x2b12be00, 0x2b3d0000)
        rw space 12288K,  46% used [0x2b3d0000, 0x2b972060, 0x2b972200, 0x2bfd0000)
    34.735: [DefNew: 52568K->3433K(55296K), 0.0072126 secs] 55264K->6615K(124928K)Heap after gc invocations=8: def new generation   total 55296K, used 3433K [0x1ebd0000, 0x227d0000, 0x227d0000)
    eden space 49152K,   0% used [0x1ebd0000, 0x1ebd0000, 0x21bd0000)
      from space 6144K,  55% used [0x21bd0000, 0x21f2a5e8, 0x221d0000)
      to   space 6144K,   0% used [0x221d0000, 0x221d0000, 0x227d0000)
     tenured generation   total 69632K, used 3182K [0x227d0000, 0x26bd0000, 0x26bd0000)
    the space 69632K,   4% used [0x227d0000, 0x22aeb958, 0x22aeba00, 0x26bd0000)
     compacting perm gen  total 8192K, used 2898K [0x26bd0000, 0x273d0000, 0x2abd0000)
       the space 8192K,  35% used [0x26bd0000, 0x26ea4ba8, 0x26ea4c00, 0x273d0000)
        ro space 8192K,  66% used [0x2abd0000, 0x2b12bcc0, 0x2b12be00, 0x2b3d0000)
        rw space 12288K,  46% used [0x2b3d0000, 0x2b972060, 0x2b972200, 0x2bfd0000)
    }
    , 0.0757599 secs]
  • -Xloggc:filename:与上面几个配合使用,把相关日志信息记录到文件以便分析。
  1. 常见配置汇总
  1. 堆设置
  • -Xms:初始堆大小
  • -Xmx:最大堆大小
  • -XX:NewSize=n:设置年轻代大小
  • -XX:NewRatio=n:设置年轻代和年老代的比值。如:为3,表示年轻代与年老代比值为1:3,年轻代占整个年轻代年老代和的1/4
  • -XX:SurvivorRatio=n:年轻代中Eden区与两个Survivor区的比值。注意Survivor区有两个。如:3,表示Eden:Survivor=3:2,一个Survivor区占整个年轻代的1/5
  • -XX:MaxPermSize=n:设置持久代大小
  1. 收集器设置
  • -XX:+UseSerialGC:设置串行收集器
  • -XX:+UseParallelGC:设置并行收集器
  • -XX:+UseParalledlOldGC:设置并行年老代收集器
  • -XX:+UseConcMarkSweepGC:设置并发收集器
  1. 垃圾回收统计信息
  • -XX:+PrintGC
  • -XX:+PrintGCDetails
  • -XX:+PrintGCTimeStamps
  • -Xloggc:filename
  1. 并行收集器设置
  • -XX:ParallelGCThreads=n:设置并行收集器收集时使用的CPU数。并行收集线程数。
  • -XX:MaxGCPauseMillis=n:设置并行收集最大暂停时间
  • -XX:GCTimeRatio=n:设置垃圾回收时间占程序运行时间的百分比。公式为1/(1+n)
  1. 并发收集器设置
  • -XX:+CMSIncrementalMode:设置为增量模式。适用于单CPU情况。
  • -XX:ParallelGCThreads=n:设置并发收集器年轻代收集方式为并行收集时,使用的CPU数。并行收集线程数。

四、调优总结

  1. 年轻代大小选择
  • 响应时间优先的应用尽可能设大,直到接近系统的最低响应时间限制(根据实际情况选择)。在此种情况下,年轻代收集发生的频率也是最小的。同时,减少到达年老代的对象。
  • 吞吐量优先的应用:尽可能的设置大,可能到达Gbit的程度。因为对响应时间没有要求,垃圾收集可以并行进行,一般适合8CPU以上的应用。
  1. 年老代大小选择
  • 响应时间优先的应用:年老代使用并发收集器,所以其大小需要小心设置,一般要考虑并发会话率会话持续时间等一些参数。如果堆设置小了,可以会造成内存碎片、高回收频率以及应用暂停而使用传统的标记清除方式;如果堆大了,则需要较长的收集时间。最优化的方案,一般需要参考以下数据获得:
  • 并发垃圾收集信息
  • 持久代并发收集次数
  • 传统GC信息
  • 花在年轻代和年老代回收上的时间比例
  • 吞吐量优先的应用:一般吞吐量优先的应用都有一个很大的年轻代和一个较小的年老代。原因是,这样可以尽可能回收掉大部分短期对象,减少中期的对象,而年老代尽存放长期存活对象。
  1. 较小堆引起的碎片问题因为年老代的并发收集器使用标记、清除算法,所以不会对堆进行压缩。当收集器回收时,他会把相邻的空间进行合并,这样可以分配给较大的对象。但是,当堆空间较小时,运行一段时间以后,就会出现“碎片”,如果并发收集器找不到足够的空间,那么并发收集器将会停止,然后使用传统的标记、清除方式进行回收。如果出现“碎片”,可能需要进行如下配置:
  • -XX:+UseCMSCompactAtFullCollection:使用并发收集器时,开启对年老代的压缩。
  • -XX:CMSFullGCsBeforeCompaction=0:上面配置开启的情况下,这里设置多少次Full GC后,对年老代进行压缩

​http://unixboy.iteye.com/blog/174173/​

JVM常用参数
-Xms 初始堆内存大小,默认物理内存64/1

-Xms = -XX:InitialHeapSize

 

-Xmx 最大堆内存,默认物理内存4/1

-Xmx = -XX:MaxHeapSize

 

-Xss 栈内存大小

设置单个线程栈大小,一般默认512~1024kb。

单个线程栈大小跟操作系统和JDK版本都有关系

-Xss = -XX:ThreadStackSize

 

-Xmn 年轻代大小

 

-XX:MetaspaceSize 元空间大小

元空间本质跟永久代类似,都是对JVM规范中方法区的实现。

不过元空间与永久代最大的区别在于:元空间并不在虚拟机中,而是使用本机内存。

因此,元空间大小仅受本地内存限制。

-XX:+PrintGCDetails 打印GC详细日志信息

-XX:SurvivorRatio 幸存者比例设置

-XX:NewRatio 新生代比例设置

-XX:MaxTenuringThreshold 进入老年代阈值设置

2、查看元空间MetaspaceSize默认值

(1)java -XX:+PrintFlagsInitial

JVM调优总结 -Xms -Xmx -Xmn -Xss_jvm_10

 

 uintx MetaspaceSize   = 21810376(大约20.80MB)    {pd product}

(2)JVM常用参数案例
【1】默认JVM命令行参数

-XX:InitialHeapSize=265858688

-XX:MaxHeapSize=4253739008

-XX:+PrintCommandLineFlags

-XX:+UseCompressedClassPointers

-XX:+UseCompressedOops

-XX:-UseLargePagesIndividualAllocation

-XX:+UseParallelGC
【2】JVM常用参数设置

-Xms128m -Xmx4096m -Xss1024k -XX:MetaspaceSize=512m -XX:+PrintCommandLineFlags -XX:+PrintGCDetails -XX:+UseSerialGC


-Djava.util.Arrays.useLegacyMergeSort=true

Array提供了两种排序算法,MergeSort和TimSort。默认用的是Java7新提供的TimSort,通过在启动参数中指定-Djava.util.Arrays.useLegacyMergeSort=true可以用原来的MergeSort。这里的TimSort不少人被一个细节坑过,异常如下:

Exception in thread "main" java.lang.IllegalArgumentException: Comparison
method violates its general contract!

 

原因是,使用TimSort时,对象间的对比较更加严格,当指定compare时,必须保证以下三点:

a). sgn(compare(x, y)) == -sgn(compare(y, x))
b). (compare(x, y)>0) && (compare(y, z)>0) 意味着 compare(x, z)>0
c). compare(x, y)==0 意味着对于任意的z:sgn(compare(x, z))==sgn(compare(y, z)) 均成立
后两点很容易满足,大多数问题都出在第一点上,例如我曾经的这个实现:

public int compare(Test o1, Test o2) {
return o1.getValue() > o2.getValue() ? 1 : -1;
}

乍一看没什么毛病,但是对比上面的条件a,就会发现不满足了,那就是:当o1的value与o2的value相等时!

解决方法:

public int compare(Test o1, Test o2) {
return o1.getValue() == o2.getValue() ? 0 : (o1.getValue() > o2.getValue() ? 1 : -1);
}

 



标签:Xss,Xmn,java,收集器,XX,调优,GC,内存,JVM
From: https://blog.51cto.com/u_15147537/5973517

相关文章