`
yunzhu
  • 浏览: 1140968 次
  • 性别: Icon_minigender_1
  • 来自: 南京
博客专栏
B2b19957-cda7-3a9e-83a0-418743feb0ca
监控应用服务器
浏览量:109081
2e8be8be-e51f-346c-bcdd-12623c9aa820
Web前端开发
浏览量:119242
Bfa5df64-a623-34b9-85b8-ef3ce2aed758
经典异常的解决
浏览量:203979
社区版块
存档分类
最新评论

异常:java.lang.OutOfMemoryError: PermGen space

阅读更多

项目中遇到这么个异常:

java.lang.OutOfMemoryError: PermGen space

 

由于项目中使用了自定义的ClassLoader,用于加载不同版本的JAR包,每启一个线程都要使用单独的ClassLoader,线程多了ClassLoader当然也就多了,于是首当其冲就怀疑很可能是由于加载class过多导致的,因为虚拟机加载的class是从来不会释放的。

 

于是请教万能的谷兄,果不其然,且看一段解释:

PermGen space的全称是Permanent Generation space,是指内存的永久保存区域,这一部分用于存放Class和Meta的信息,Class在被 Load的时候被放入PermGen space区域,它和和存放Instance的Heap区域不同,GC(Garbage Collection)不会在主程序运行期对PermGen space进行清理,所以如果你的APP会LOAD很多CLASS的话,就很可能出现PermGen space错误。

 

原来,正是由于每个ClassLoader加载的class都留在了PermGen space这个区域,而启的线程越多,加载的class也越多,造成很多class被重复加载,于是对PermGen space的占用呈几何级增长,溢出成为必然。

 

症结找到了,下面就要解决了:每个线程使用的ClassLoader并非完全没有雷同,实际上所有线程使用的ClassLoader也就那么几个是不同的,其他的都是这个几个的“拷贝”,因此,只要将这几个不同的ClassLoader实现复用,不就OK了吗?是的,然后同类型的线程使用同一个ClassLoader就行了,这样就不会重复加载相同的class了,问题解决了。

 

 

2
6
分享到:
评论
7 楼 yunzhu 2014-04-14  
lvwenwen 写道
同类型的线程加载A.jar,B.jar,这个同类型怎么理解,怎么控制线程不重复加载,请教下,谢谢

并入多个线程做的是同样的工作,比如监控WebLogic,需要用到weblogic.jar,那么这些线程只使用同一个类加载,不能每个线程使用单独的类加载器,也不需要担心同步问题
6 楼 lvwenwen 2014-04-13  
同类型的线程加载A.jar,B.jar,这个同类型怎么理解,怎么控制线程不重复加载,请教下,谢谢
5 楼 yunzhu 2013-12-23  
aspnetdb 写道
yunzhu 写道
buru 写道
”,因此,只要将这几个不同的ClassLoader实现复用,不就OK了吗?是的,然后同类型的线程使用同一个ClassLoader就行了,这样就不会重复加载相同的class了,问题解决了。


具体怎么做的,能详细说说吗,多谢!


就是我有一些线程使用自定义ClassLoader加载A.jar,有一些线程使用自定义ClassLoader加载B.jar,如果每个线程使用一个ClassLoader,那么A.jar和B.jar中的class就会被重复加载N次,就造成了文中所说的OutOfMemoryError错误,因此,我将加载A.jar的ClassLoader进行复用,将加载B.jar的ClassLoader也进行复用,A.jar和B.jar中的class就不会被重复加载了,就不会发生OutOfMemoryError错误了


那A.jar和B.jar假如有类冲突了怎么办?

我这里就是因为A.jar和B.jar的类冲突,还有A.jar跟JDK的类,所以才实现了自定义的类加载器
4 楼 aspnetdb 2013-12-23  
yunzhu 写道
buru 写道
”,因此,只要将这几个不同的ClassLoader实现复用,不就OK了吗?是的,然后同类型的线程使用同一个ClassLoader就行了,这样就不会重复加载相同的class了,问题解决了。


具体怎么做的,能详细说说吗,多谢!


就是我有一些线程使用自定义ClassLoader加载A.jar,有一些线程使用自定义ClassLoader加载B.jar,如果每个线程使用一个ClassLoader,那么A.jar和B.jar中的class就会被重复加载N次,就造成了文中所说的OutOfMemoryError错误,因此,我将加载A.jar的ClassLoader进行复用,将加载B.jar的ClassLoader也进行复用,A.jar和B.jar中的class就不会被重复加载了,就不会发生OutOfMemoryError错误了


那A.jar和B.jar假如有类冲突了怎么办?
3 楼 zhangli123123 2013-03-20  
2 楼 yunzhu 2011-05-09  
buru 写道
”,因此,只要将这几个不同的ClassLoader实现复用,不就OK了吗?是的,然后同类型的线程使用同一个ClassLoader就行了,这样就不会重复加载相同的class了,问题解决了。


具体怎么做的,能详细说说吗,多谢!


就是我有一些线程使用自定义ClassLoader加载A.jar,有一些线程使用自定义ClassLoader加载B.jar,如果每个线程使用一个ClassLoader,那么A.jar和B.jar中的class就会被重复加载N次,就造成了文中所说的OutOfMemoryError错误,因此,我将加载A.jar的ClassLoader进行复用,将加载B.jar的ClassLoader也进行复用,A.jar和B.jar中的class就不会被重复加载了,就不会发生OutOfMemoryError错误了
1 楼 buru 2011-05-07  
”,因此,只要将这几个不同的ClassLoader实现复用,不就OK了吗?是的,然后同类型的线程使用同一个ClassLoader就行了,这样就不会重复加载相同的class了,问题解决了。


具体怎么做的,能详细说说吗,多谢!

相关推荐

Global site tag (gtag.js) - Google Analytics