java中的对象都创建在堆中,当堆中的对象所占空间到一定量后,就需要清除掉死亡对象,垃圾回收是JVM自动完成的,不用码农手动清理,但偶尔想看看对象所占的堆空间或GC的具体过程,就需要借助GC日志了。
一、打印GC日志
1、示例程序
程序1
public class GCTest {
public static void main(String[] args) {
for (int i = 0; i < 10; i++) {
byte[] b = new byte[1*1024*1024];
}
}
}
2、加VM参数 -XX:+PrintGCDetail(以idea为例)
data:image/s3,"s3://crabby-images/556a9/556a9ff88221bd1412254097aac5de0a9019bb02" alt=""
data:image/s3,"s3://crabby-images/27e78/27e78e76c1eaaea64b8b3989cea16b5043f0a5fa" alt=""
最后记得点击下面的apply和ok,然后运行GCTest,就可以看到:对象都放在eden区48%(10M)的空间,目前没有发生GC,所以Survivor区和年老代都是0%。
data:image/s3,"s3://crabby-images/0bef2/0bef2f46b8ce2183c230c5bdc32313a0f9996ca2" alt=""
3、调整GCTest创建的对象大小为30M
程序2
public class GCTest {
public static void main(String[] args) {
for (int i = 0; i < 30; i++) {
byte[] b = new byte[1*1024*1024];
}
}
}
再次运行,可以看到发生了一次GC,将幸存的对象移至Survivor from区,所以eden、from都有对象存在了。
data:image/s3,"s3://crabby-images/63cb5/63cb52dfba07d051ac65f99c19bc7a7d9cf2638e" alt=""
GC日志:[GC (Allocation Failure) [PSYoungGen: 32978K->1512K(38400K)] 32978K->1520K(125952K), 0.0020363 secs] [Times: user=0.00 sys=0.00, real=0.00 secs]
可以看出新生代 YoungGen 分配失败,32978K->1512K(38400K) 表示 GC 前已使用->GC后已使用(该区域总容量),[]之外表示Java堆GC前已使用->GC后已使用(Java堆总容量),最后为 GC 耗时。
4、-XX:+PrintHeapAtGC
-XX:+PrintHeapAtGC的含义是打印GC前后的详细堆栈信息
将它配置在VM options中,运行程序2,就可以看到:
data:image/s3,"s3://crabby-images/346a1/346a19ba57ea32e4b410b34d9882226d5990e186" alt=""
5、-XX:+TraceClassLoading
-XX:+TraceClassLoading的作用是监控类的加载
将它配置在VM options中,运行程序2,就可以看到:
将它配置在VM参数中,运行程序2,就可以看到:
data:image/s3,"s3://crabby-images/08656/08656988b5a2a83cf3618f73ac691708cc159c89" alt=""
二、堆分配参数
程序3
public class TestMaxMinHeap {
private static final int M = 1024 * 1024;
public static void main(String[] args) {
System.out.println("maxMemory:" + ( Runtime.getRuntime().maxMemory()/M) + "M");
System.out.println("freeMemory:" + (Runtime.getRuntime().freeMemory()/M) + "M");
System.out.println("totalMemory:" + (Runtime.getRuntime().totalMemory()/M) + "M");
}
}
不加任何参数的时候,运行结果是:
data:image/s3,"s3://crabby-images/f17ad/f17ad12866ebb1642537bac25bfd85b0cce226fc" alt=""
1、-Xmx 和 -Xms
-Xmx是最大堆大小,默认值是物理内存的1/4(<1GB),
-Xmx20m 设置 maxMemory 为 20 M,分配空间不能超过最大堆内存大小,否则会抛出 OutofMemory 异常
data:image/s3,"s3://crabby-images/27761/277614002c72dcc1369fdf527b3350025a7aca4b" alt=""
-Xms是初始堆大小,默认值是物理内存的1/64(<1GB),-Xms20m 设置 total Memory 为 20 M ,totalMemory 就是初始化堆大小,它的意思是一开始限定堆大小为多少,如果不够则可以扩充,但必须小于最大堆大小。
data:image/s3,"s3://crabby-images/497a4/497a49d9f230065cf36d24cc276c65f2ca6b54d8" alt=""
2、测试动态调整
①首先设置 -Xmx20m -Xms5m ,最大堆空间20m,初始大小5m。
data:image/s3,"s3://crabby-images/a3a98/a3a98879f4d4c06841e949ecd1362e8349af06fb" alt=""
②分配2M
程序4
public class TestMaxMinHeap {
private static final int M = 1024 * 1024;
public static void main(String[] args) {
byte b[] = new byte[2 * M];
System.out.println("maxMemory:" + ( Runtime.getRuntime().maxMemory()/M) + "M");
System.out.println("freeMemory:" + (Runtime.getRuntime().freeMemory()/M) + "M");
System.out.println("totalMemory:" + (Runtime.getRuntime().totalMemory()/M) + "M");
}
}
data:image/s3,"s3://crabby-images/7237b/7237bc28aed61036ff2818b6b6c341487b6fe09e" alt=""
③分配6M
程序5
public class TestMaxMinHeap {
private static final int M = 1024 * 1024;
public static void main(String[] args) {
byte b[] = new byte[6 * M];
System.out.println("maxMemory:" + ( Runtime.getRuntime().maxMemory()/M) + "M");
System.out.println("freeMemory:" + (Runtime.getRuntime().freeMemory()/M) + "M");
System.out.println("totalMemory:" + (Runtime.getRuntime().totalMemory()/M) + "M");
}
}
data:image/s3,"s3://crabby-images/15600/15600bd2b06ecc9da59445685daccfeb7b1907a4" alt=""
③分配30M
程序6
public class TestMaxMinHeap {
private static final int M = 1024 * 1024;
public static void main(String[] args) {
byte b[] = new byte[30 * M];
System.out.println("maxMemory:" + ( Runtime.getRuntime().maxMemory()/M) + "M");
System.out.println("freeMemory:" + (Runtime.getRuntime().freeMemory()/M) + "M");
System.out.println("totalMemory:" + (Runtime.getRuntime().totalMemory()/M) + "M");
}
}
data:image/s3,"s3://crabby-images/89137/89137014ea6ca2db5cc629386cdac0c3fb147876" alt=""
2、-Xmn
-Xmn:设置 新生代 大小
①新生代大小设置为 2M,-Xmn2m -Xms20m -Xmx20m -XX:+PrintGCDetails
程序7
public class TestNewSpace {
private static final int M = 1024*1024;
public static void main(String[] args) {
byte[] b = null;
for (int i = 0; i < 10; ++i) {
b = new byte[1*M];
}
}
}
data:image/s3,"s3://crabby-images/a8c91/a8c91e22abed3eff6c85e5305ac0479362e3504c" alt=""
②设置新生代为18M,-Xmn18m -Xms20m -Xmx20m -XX:+PrintGCDetails
data:image/s3,"s3://crabby-images/b356f/b356f512d2d55365cfa52f9ea0fa87b85dad79d7" alt=""
3、-SurvivorRatio
-SurvivorRatio:设置新生代中 Eden space 和 Survivor space 的大小
①设置新生代8m,-Xmn8m -Xms20m -Xmx20m -XX:+PrintGCDetails
data:image/s3,"s3://crabby-images/1f151/1f151c3cd7ec62b15faf0f8587b6dca60fa84a0c" alt=""
②将新生代中的的Survivor设置大一点,-Xmn8m -Xms20m -Xmx20m -XX:+PrintGCDetails -XX:SurvivorRatio=2
data:image/s3,"s3://crabby-images/4ed74/4ed74bc9c26284ec1fff971a6957cf15deca7f0e" alt=""
4、-XX:NewRatio
-XX:NewRatio 设置年老代和年轻代的比值
①设置年老代和年轻代的比例是2:1
-Xms20m -Xmx20m -XX:+PrintGCDetails -XX:NewRatio=2
data:image/s3,"s3://crabby-images/2ee8b/2ee8b66c9c0d5a40e183b210853524018977be5b" alt=""
②设置年老代和年轻代的比例是3:1
-Xms20m -Xmx20m -XX:+PrintGCDetails -XX:NewRatio=3
data:image/s3,"s3://crabby-images/32f0c/32f0c883c06cbc04ddfa9f1bd39595642c940c73" alt=""
5、-Xss
-Xss设置每个线程的堆栈大小
①设置线程栈为128K,-Xss128k
程序8
public class TestXssStack {
public static int count = 0;
public static void func() {
count++;
func();
}
public static void main(String[] args) {
try {
func();
}catch (Throwable e){
System.out.println("count:" + count);
}
}
}
data:image/s3,"s3://crabby-images/bd62b/bd62b6ce16c54abfdea09de9302c726d62b2a5b6" alt=""
②设置线程栈为1m,-Xss1m
data:image/s3,"s3://crabby-images/c4d26/c4d26658a648819acf85e8a6eddb26a0c5ccbccd" alt=""
总结
本篇主要简单介绍了如何使用idea查看GC日志,以及JVM参数-XX:+PrintGCDetail、-XX:+PrintHeapAtGC、-XX:+TraceClassLoading、-Xmx 、-Xms、-Xmn、-SurvivorRatio、-XX:NewRatio、-Xss所代表的意义和使用。
网友评论