什么是OutOfMemoryError
OutOfMemoryError就是我们java语言中的内存溢出。
简单说下内存溢出和与之相关的内存泄露
内存泄露和内存溢出的关系:
总结:内存泄露导致内存溢出
内存泄露
内存泄露的原因
通俗的讲,内存泄露就是我们申请的内存空间没有被回收。
当我们创建一个对象,而在操作完成后没有对其进行处理,导致垃圾回收器没有及时进行回收,会造成短暂的内存泄露。
短暂的内存泄漏并不会对系统造成多大影响,但这只是基于一个对象的情况,实际开发中,循环,递归等操作经常创造多个对象,如果没有及时进行内存空间的回收,因为JVM默认的内存空间大小事有限的(此处不涉及自定义内存大小),多次内存泄露会导致我们无法为新的对象申请足够的内存空间,程序到这就会报错。
例子:
举个内存泄露例子:
class A{
Object obj;
public void showObj(){
obj=new Object();
}
....
}
此处,Object对象随着showObj方法的结束已经没有了用处,但因为obj引用指向着该内存地址,gc无法马上处理这个没用的对象,严格上说这也是一种内存泄漏,从系统优化上可以这样:
class A{
Object obj;
public void showObj(){
obj=new Object();
.....
obj=null;
}
}
此时gc能快速将已创建的Object标记,当下次执行回收操作时将这个没用的对象回收,重新腾出内存空间。
上述表面上只创建了一个对象,有同学可能会想,只是一个对象内存空间而已不会有多大影响,但这体现了我们对系统的代码的优化的理解性。再举一个创建数组对象的例子
class A {
List list;
public void addList() {
for (int i = 0; i <= 10; i++) {
list = new ArrayList();
list.add(new Object());
}
for (Object i :list) {
i=null;
}
}
}
执行到此处,所有的Object对象都是null,然而随着addList()方法的结束,这些null对象对没有被垃圾回收器回收,因为有一个长生命周期的引用指向他们,就是list引用,和尚一个例子相比,此处产生的无法回收对象更多,所以,代码的优化是很有必要的。解决上述内存泄露问题只需要在addList()方法末尾加上 list=null;
举个简单内存溢出例子,如果程序陷入死循环,创建新的对象
class A{
public A(){
new A();
}
}
因为不断创建对象的原因,开始和中途创建的对象因为程序没有结束,无法被Gc垃圾回收器标记为回收,产生的内存对象会不断占用内存,最终导致没有足够的内存空间来存放新对象,导致内存溢出。
内存溢出的三种情况
一、OutOfMemoryError: PermGen space
这种情况表明JVM的永久保存区域(方法区),没有足够的空间存放class文件或jar包,原因是程序中使用了大量的Jar包和class文件,导致JVM装载类的空间不够。
这种情况有两种解决方法:
1. 增加java虚拟机中的XX:PermSize和XX:MaxPermSize参数的大小,其中XX:PermSize是初始永久保存区域大小,XX:MaxPermSize是最大永久保存区域大小。
2. 清理应用程序中web-inf/lib下的jar,如果tomcat部署了多个应用,很多应用都使用了相同的jar,可以将共同的jar移到tomcat共同的lib下,减少类的重复加载。
二、OutOfMemoryError: Java heap space
以上举例说明的内存泄露都是产生该种内存溢出的原因,各位学习编程的时候应注意不要过多创建对象,对于没有用的对象应及时将其设置为null,交给垃圾回收器回收。
一般有两种解决思路:
1.检查程序,包括死循环和多对象创建但没有及时处理的地方,防止虚拟机的堆内存空间占满。改进程序,提高代码的执行效率。
2.增加Java虚拟机中Xms(初始堆大小)和Xmx(最大堆大小)参数的大小。
三、OutOfMemoryError:unable to create new native thread
参考文章: