GC overhead limit exceeded填坑心得
我遇到这样的问题,本地部署时抛出异常java.lang.OutOfMemoryError:GC overhead limit exceeded导致服务起不来,查看日志发现加载了太多资源到内存,本地的性能也不好,gc时间消耗的较多。解决这种问题两种方法是,增加参数,-XX:-UseGCOverheadLimit,关闭这个特性,同时增加heap大小,-Xmx1024m。坑填了,but why?
OOM大家都知道,就是JVM内存溢出了,那GC overhead limit exceed呢?
GC overhead limt exceed检查是Hotspot VM 1.6定义的一个策略,通过统计GC时间来预测是否要OOM了,提前抛出异常,防止OOM发生。Sun 官方对此的定义是:“并行/并发回收器在GC回收时间过长时会抛出OutOfMemroyError。过长的定义是,超过98%的时间用来做GC并且回收了不到2%的堆内存。用来避免内存过小造成应用不能正常工作。“
听起来没啥用...预测OOM有啥用?起初开来这玩意只能用来Catch住释放内存资源,避免应用挂掉。后来发现一般情况下这个策略不能拯救你的应用,但是可以在应用挂掉之前做最后的挣扎,比如数据保存或者保存现场(Heap Dump)。
而且有些时候这个策略还会带来问题,比如加载某个大的内存数据时频繁OOM。
假如你也生产环境中遇到了这个问题,在不知道原因时不要简单的猜测和规避。可以通过-verbose:gc -XX:+PrintGCDetails看下到底什么原因造成了异常。通常原因都是因为old区占用过多导致频繁Full GC,最终导致GC overhead limit exceed。如果gc log不够可以借助于JProfile等工具查看内存的占用,old区是否有内存泄露。分析内存泄露还有一个方法-XX:+HeapDumpOnOutOfMemoryError,这样OOM时会自动做Heap Dump,可以拿MAT来排查了。还要留意young区,如果有过多短暂对象分配,可能也会抛这个异常。
日志的信息不难理解,就是每次gc时打条日志,记录GC的类型,前后大小和时间。举个例子。
33.125: [GC [DefNew: 16000K->16000K(16192K), 0.0000574 secs][Tenured: 2973K->2704K(16384K), 0.1012650 secs] 18973K->2704K(32576K), 0.1015066 secs]
100.667:[Full GC [Tenured: 0K->210K(10240K), 0.0149142 secs] 4603K->210K(19456K), [Perm : 2999K->2999K(21248K)], 0.0150007 secs]
GC和Full GC代表gc的停顿类型,Full GC代表stop-the-world。箭头两边是gc前后的区空间大小,分别是young区、tenured区和perm区,括号里是该区的总大小。冒号前面是gc发生的时间,单位是秒,从jvm启动开始计算。DefNew代表Serial收集器,为Default New Generation的缩写,类似的还有PSYoungGen,代表Parallel Scavenge收集器。这样可以通过分析日志找到导致GC overhead limit exceeded的原因,通过调节相应的参数解决问题。
文中涉及到的名词解释,
Eden Space:堆内存池,大多数对象在这里分配内存空间。
Survivor Space:堆内存池,存储在Eden Space的gc中存活下来的对象。
Tenured Generation:堆内存池,存储Survivor Space中存活过几次gc的对象。
Permanent Generation:非堆空间,存储的是class和method对象。
Code Cache:非堆空间,JVM用来存储编译和存储native code。
最后附上GC overhead limit exceed HotSpot的实现:
| 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 | boolprint_gc_overhead_limit_would_be_exceeded = false;if(is_full_gc) {  if(gc_cost() > gc_cost_limit &&    free_in_old_gen < (size_t) mem_free_old_limit &&    free_in_eden < (size_t) mem_free_eden_limit) {    // Collections, on average, are taking too much time, and    //      gc_cost() > gc_cost_limit    // we have too little space available after a full gc.    //      total_free_limit < mem_free_limit    // where    //   total_free_limit is the free space available in    //     both generations    //   total_mem is the total space available for allocation    //     in both generations (survivor spaces are not included    //     just as they are not included in eden_limit).    //   mem_free_limit is a fraction of total_mem judged to be an    //     acceptable amount that is still unused.    // The heap can ask for the value of this variable when deciding    // whether to thrown an OutOfMemory error.    // Note that the gc time limit test only works for the collections    // of the young gen + tenured gen and not for collections of the    // permanent gen.  That is because the calculation of the space    // freed by the collection is the free space in the young gen +    // tenured gen.    // At this point the GC overhead limit is being exceeded.    inc_gc_overhead_limit_count();    if(UseGCOverheadLimit) {      if(gc_overhead_limit_count() >=          AdaptiveSizePolicyGCTimeLimitThreshold){        // All conditions have been met for throwing an out-of-memory        set_gc_overhead_limit_exceeded(true);        // Avoid consecutive OOM due to the gc time limit by resetting        // the counter.        reset_gc_overhead_limit_count();      } else{        // The required consecutive collections which exceed the        // GC time limit may or may not have been reached. We        // are approaching that condition and so as not to        // throw an out-of-memory before all SoftRef's have been        // cleared, set _should_clear_all_soft_refs in CollectorPolicy.        // The clearing will be done on the next GC.        boolnear_limit = gc_overhead_limit_near();        if(near_limit) {          collector_policy->set_should_clear_all_soft_refs(true);          if(PrintGCDetails && Verbose) {            gclog_or_tty->print_cr("  Nearing GC overhead limit, "              "will be clearing all SoftReference");          }        }      }    }    // Set this even when the overhead limit will not    // cause an out-of-memory.  Diagnostic message indicating    // that the overhead limit is being exceeded is sometimes    // printed.    print_gc_overhead_limit_would_be_exceeded = true;  } else{    // Did not exceed overhead limits    reset_gc_overhead_limit_count();  }} | 
参照&延伸阅读:
http://javaeesupportpatterns.blogspot.com/2012/01/gc-overhead-limit-exceeded-understand.html
http://www.oracle.com/technetwork/java/javase/gc-tuning-6-140523.html
http://reins.altervista.org/java/gc1.4.2_example.html
http://stackoverflow.com/questions/2129044/java-heap-terminology-young-old-and-permanent-generations
http://book.51cto.com/art/201306/399236.htm
https://blogs.oracle.com/jonthecollector/entry/presenting_the_permanent_generation
GC overhead limit exceeded填坑心得的更多相关文章
- java.lang.OutOfMemoryError:GC overhead limit exceeded填坑心得
		我遇到这样的问题,本地部署时抛出异常java.lang.OutOfMemoryError:GC overhead limit exceeded导致服务起不来,查看日志发现加载了太多资源到内存,本地的性 ... 
- [转]java.lang.OutOfMemoryError:GC overhead limit exceeded
		我遇到这样的问题,本地部署时抛出异常java.lang.OutOfMemoryError:GC overhead limit exceeded导致服务起不来,查看日志发现加载了太多资源到内存,本地的性 ... 
- 【OOM】GC overhead limit exceeded
		我遇到这样的问题,本地部署时抛出异常java.lang.OutOfMemoryError:GC overhead limit exceeded导致服务起不来,查看日志发现加载了太多资源到内存,本地的性 ... 
- 转载:OutOfMemoryError系列(2): GC overhead limit exceeded
		这是本系列的第二篇文章, 相关文章列表: OutOfMemoryError系列(1): Java heap space OutOfMemoryError系列(2): GC overhead limit ... 
- java.lang.OutOfMemoryError GC overhead limit exceeded原因分析及解决方案
		最近一个上线运行良好的项目出现用户无法登录或者执行某个操作时,有卡顿现象.查看了日志,出现了大量的java.lang.OutOfMemoryError: GC overhead limit excee ... 
- OutOfMemoryError系列(2): GC overhead limit exceeded
		原因分析 JVM抛出 java.lang.OutOfMemoryError: GC overhead limit exceeded 错误就是发出了这样的信号: 执行垃圾收集的时间比例太大, 有效的运算 ... 
- Spark java.lang.outofmemoryerror gc overhead limit exceeded 与 spark OOM:java heap space 解决方法
		引用自:http://cache.baiducontent.com/c?m=9f65cb4a8c8507ed4fece7631046893b4c4380146d96864968d4e414c42246 ... 
- Unable to execute dex: GC overhead limit exceeded
		Android打包时下面的错误: Unable to execute dex: GC overhead limit exceeded GC overhead limit exceeded 解决的方法: ... 
- android studio Error:java.lang.OutOfMemoryError: GC overhead limit exceeded
		android studio Error:java.lang.OutOfMemoryError: GC overhead limit exceeded 在app下的build.gradle中找到and ... 
随机推荐
- PHP多台服务器跨域SESSION共享
			网站业务规模和访问量的逐步发展,原本由单台服务器.单个域名的迷你网站架构已经无法满足发展需要. 此时我们可能会购买更多服务器,并且启用多个二级子域名以频道化的方式,根据业务功能将网站分布部署在独立的服 ... 
- 【Android测试】【第十七节】Instrumentation——App任你摆布(反射技术的引入)
			◆版权声明:本文出自胖喵~的博客,转载必须注明出处. 转载请注明出处:http://www.cnblogs.com/by-dream/p/5569844.html 前言 学习了上节之后,大家是否已经 ... 
- centos升级支持到C++11, gcc4.8.2
			升级到4.8[这个应该是目前最新的啦,不过网上查的话已经到5.2啦,感觉落后一点比较稳,当然还有就是这个版本是新的里面使用最多的] wget http://people.centos.org/tru/ ... 
- 转:"在已损坏了程序内部状态的XXX.exe 中发生了缓冲区溢出"的一种可能原因
			我的问题跟原作者的问题差不多.头文件和DLL不匹配导致的. 原文链接:http://blog.csdn.net/u012494876/article/details/39030887 今天软件突然出现 ... 
- 【转】PHP实现连贯操作
			[第一种方案 __call] 我们在使用一些框架(如ThinkPHP)编码的时候,常用到这样的代码. M('User')->where(array('id'=>1))->field( ... 
- RDIFramework.NET ━ Web中打印的各种方案参考-欢迎补充
			RDIFramework.NET ━ Web中打印的各种方案参考-欢迎补充 做Web开发的同志应该都深有体会,在web程序中打印不再象应用程序中那样便于控制了,web程序天生的一些特性造成了这个缺点, ... 
- 一款bootstrap树形js
			http://www.htmleaf.com/Demo/201502141380.html 
- 清除div中内容
			$.ajax({ url: "SearchSN.aspx", data: "SN=" + $("#txtS ... 
- 复制  replication  新增表 add new article
			immediate_sync false Synchronization files are created only if there are new subscriptions. Subs ... 
- ListView的LayoutParams设置
			// Temp is the root view that was found in the xml final View temp = createViewFromTag(root, name, a ... 
