深入JVM系列 – 配置参数

JVM的配置参数分为三类:

跟踪参数

堆分配参数

栈分配参数

 

跟踪参数

 

跟踪参数用于跟踪监控JVM,往往被开发人员用于JVM调优以及故障排查。

gc发生时,打印GC的简要信息

-XX:+PrintGC或-verbose:gc 这两个配置的效果一样,都是在GC发生时打印出简要的信息,执行以下代码:

public static void main(String[] args){

   byte[] bytes = null;

   for(int i=0; i<100; i++){

        bytes = new byte[1024*1024];

   }

}

该程序连续创建100个1M的数组, 使用上面两个参数中的一个就可以看到GC的情况:

151621_9YVu_3295294.png

可以看到程序执行了一次GC(新生代GC), 因为每次创建新的数组都会赋值给bytes,导致老的数组没有对象引用,因此老对象会变为不可达状态,这些对象就会被回收掉。其中511K

表示回收前对象占用的内存;472K表示回收后对象占用的内存;163328K表示还有多少内存可用; 0.0398984 secs表示该粗垃圾回收花费的时间。

打印GC的详细信息以及堆使用详细信息

 -XX:+PrintGCDetails

依然使用上面的程序查看GC情况:

151648_ucgc_3295294.png

除了看到GC的信息外,还显示了堆内存的使用情况, 堆分为新生代、老年代、元空间,注意java8中移除了永久区,原来放在永久区的常量、字符串静态变量都移动到了元空间,并使用本地内存。

新生代又分为伊甸区(eden)和幸存区(from和to),从上面的打印可以看出新生代总大小为1024K,使用了710K。细心的同学的可能会发现eden+from+to=512K+512K+512K=1536K 并不等于总大小1024K,这是为什么呢?这是因为新生代的垃圾回收算法是采用复制算法,简单的说就是在from和to之间来回复制(复制过程中再把不可达的对象回收掉),所以必须保证其中一个区是空的,这样才能有预留空间存放复制过来的数据,所以新生代的总大小其实等于eden+from(或to)=512K+512K =1024K。

使用外部文件记录GC

它可以把GC的日志记录到外部文件中,这在生产环境进行故障排查时尤为重要,当java程序出现OOM时,总希望看到当时垃圾回收的情况,通过这个参数就可以把GC的日志记录下来,便于排查问题,当然也可以做日常JVM监控。

-Xloggc:log/gc.log

依然使用上面的程序查看GC情况:

151734_oMtN_3295294.png

监控类的加载

-XX:+TraceClassLoading

使用该参数可以监控java程序加载的类

151801_T0uZ_3295294.png

 

堆配置参数

 

指定最大堆,最小堆,-Xmx、-Xms

执行以下程序可以打印当前内存的使用情况

public static void main(String[] args){

  System.out.println(“Max Heap:” + Runtime.getRuntime().maxMemory()/1024/1024 + “M”);

  System.out.println(“Free Heap:” + Runtime.getRuntime().freeMemory()/1024/1024 + “M”);

  System.out.println(“Total Heap:” + Runtime.getRuntime().totalMemory()/1024/1024 + “M”);

}

最大堆:Xmx指定的大小,表示java程序最大能使用的内存大小,内存使用超过这个值,就会报:out of memory.

空闲堆: 表示程序已经分配的内存大小减去已经使用的内存大小。

总的堆:表示目前程序已经配置到多少内存大小,一般程序启动,会按照-Xms5m先分配5M的空间,这是总的堆大小就是5M。

指定新生代内存大小,-Xmn

继续执行上面的程序并指定-Xmx20m -Xms5m -Xmn2m -XX:+PrintGCDetails 参数

151823_fIcL_3295294.png

可以看出eden+from+to 等于-Xmn指定的值。

指定新生代和老年代的比值,-XX:NewRatio[新生代:年老代]

继续执行上面的程序并指定-Xmx20m –Xms20m –XX:NewRatio=4 -XX:+PrintGCDetails 参数

151924_x9Oa_3295294.png

可以看到新生代(eden+from+to=4096k)和年老代(16384k-不包含永久区)的比值4096:16384 = 1:4和–XX:NewRatio=4相吻合。

指定Eden和Survivor区(from+to)的比值,-XX:SurvivorRatio(如果设置为8,则两个Survivor区与一个Eden区的比值为2:8, 一个Survivor区占整个年轻代的1/10)

继续执行上面的程序并指定-Xmx20m –Xms20m –Xmn8m –XX:SurvivorRatio=6 -XX:+PrintGCDetails 参数

Survivor = from + to = 2048k, eden = 6144k,  survivor:eden = 2:6与–XX:SurvivorRatio=6吻合。

 

-XX:+HeapDumpOnOutOfMemoryError,-XX:+HeapDumpPath这两个参数可以实现在发生OOM异常时把堆栈信息打印到外部文件。

使用方法: -XX:+HeapDumpOnOutOfMemoryError -XX:HeapDumpPath=D:\gc_study\log_hprof\gc.hprof。然后可以参考jhat命令查看堆栈信息。

 

堆分配参数总结

 

根据实际情况调整新生代和幸存代的大小

官方推荐新生代占堆的3/8

幸存区占新生代的1/10

在OOM时,记得Dump出堆,确保可以排查现场问题

 

永久区分配参数【jdk1.8+已废弃】

-XX:PermSize -XX:MaxPermSize

用于设置永久区的初始空间和最大空间,他们表示一个系统可以容纳多少个类型,一般空间比较小。由于java8后,永久区被移动到了元数据库,使用本地内存,因此这两个参数也不建议使用了

栈大小分配参数

栈大小参数为-Xss,通常只有几百k,决定了函数调用的深度,每个线程都有自己独立的栈空间。如果函数调用太深,超过了栈的大小,则会抛出java.lang.StackOverflowError,通常我们遇到这种错误,不是去调整-Xss参数,而是应该去调查函数调用太深的原理,是否使用递归,能不能保证递归出口等。

 

转载于:https://my.oschina.net/javastorm/blog/1789901

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值