java字段太多会栈溢出_Java内存溢出与栈溢出

https://blog.csdn.net/z69183787/article/details/75530650

一、背景知识

1、JVM体系结构

eeee516767381bfc61d2dc7773621898.png

2、JVM运行时数据区

9c6208cf719290765b7573ddd69e3409.png

3、JVM内存模型

JVM运行时内存 = 共享内存区 + 线程内存区

cb3c4f3f5f82ed2ed84670ea431c116d.png

3-1、共享内存区

共享内存区 = 持久带 + 堆

持久带 = 方法区 + 其他

堆 = Old Space + Young Space

Young Space = Eden + S0 + S1

a534a81c4d77da639bc41a35fe06c8d6.png

3-1-1、持久代

JVM用持久带(Permanent Space)实现方法区,主要存放所有已加载的类信息,方法信息,常量池等等。

可通过-XX:PermSize和-XX:MaxPermSize来指定持久带初始化值和最大值。

Permanent Space并不等同于方法区,只不过是Hotspot JVM用Permanent Space来实现方法区而已,有些虚拟机没

有Permanent Space而用其他机制来实现方法区。

3-1-2、堆

堆(heap),主要用来存放类的对象实例信息(包括new操作实例化的对象和定义的数组)。

堆分为Old Space(又名,Tenured Generation)和Young Space。

Old Space主要存放应用程序中生命周期长的存活对象;

Eden(伊甸园)主要存放新生的对象;

S0和S1是两个大小相同的内存区域,主要存放每次垃圾回收后Eden存活的对象,作为对象从Eden过渡到Old Space

的缓冲地带(S是指英文单词Survivor Space)。

堆之所以要划分区间,是为了方便对象创建和垃圾回收,后面垃圾回收部分会解释。

3-2、线程内存区

线程内存区=单个线程内存+单个线程内存+.......

单个线程内存=PC Regster+JVM栈+本地方法栈

JVM栈=栈帧+栈帧+.....

栈帧=局域变量区+操作数区+帧数据区

f2d235136d019d4d77f4e2efd2b2aa3c.png

在Java中,一个线程会对应一个JVM栈(JVM Stack),JVM栈里记录了线程的运行状态。

JVM栈以栈帧为单位组成,一个栈帧代表一个方法调用。栈帧由三部分组成:局部变量区、操作数栈、帧数据区。

二、堆溢出

堆(Heap)是Java存放对象实例的地方。

堆溢出可以分为以下两种情况,这两种情况都会抛出OutOfMemoryError:java heap space异常:

1、内存泄漏

内存泄漏是指对象实例在新建和使用完毕后,仍然被引用,没能被垃圾回收释放,一直积累,直到没有剩余

内存可用。

如果内存泄露,我们要找出泄露的对象是怎么被GC ROOT引用起来,然后通过引用链来具体分析泄露的原因。

分析内存泄漏的工具有:Jprofiler,visualvm等。

示例代码:

packagecom.jvm;importjava.util.ArrayList;importjava.util.List;importjava.util.UUID;/*** 内存泄漏

*@authorfeizi

* @time 2015-1-23上午8:42:53*/

public classOOMTest {public static voidmain(String[] args) {

List list = new ArrayList();while(true){

list.add(UUID.randomUUID());

}

}

}

看看控制台的输出结果,因为我这边的JVM设置的参数内存足够大,所以需要等待一定的时间,才能看到效果:

99e090ba5cc16acc969bd1b74c75be10.png

如果是用CMD命令行,就可以自己指定参数编译运行了,这样效果就更快一些:

通过下列命令运行程序,注意先要用javac命令将.java源文件编译成.class类字节码文件。

java -Xms10M -Xmx10M -XX:-UseGCOverheadLimit OOMTest

60a1b20e1fd4cf1ff811c86ec8b7b3e5.png

2、内存溢出

内存溢出是指当我们新建一个实力对象时,实例对象所需占用的内存空间大于堆的可用空间。

如果出现了内存溢出问题,这往往是程序本生需要的内存大于了我们给虚拟机配置的内存,这种情况下,我们可以采用调大-Xmx来解决这种问题。

示例代码:

packagecom.jvm;importjava.util.ArrayList;importjava.util.List;/*** 内存溢出

*@authorfeizi

* @time 2015-1-23上午8:56:22*/

public classOOMTest_1 {public static voidmain(String args[]){

List byteList = new ArrayList();

byteList.add(new byte[1000 * 1024 * 1024]);

}

}

看看控制台的运行效果:

84bfed3ca4a9d4bda95ac7f2255571f6.png

使用CMD命令行指定参数运行:

java -verbose:gc -Xmn10M -Xms20M -Xmx20M -XX:+PrintGC OOMTest_1

d60bdb8ff59641cb67d314f98b535574.png

三、线程栈

栈(JVM Stack)存放主要是栈帧( 局部变量表, 操作数栈 , 动态链接 , 方法出口信息 )的地方。注意区分栈和栈帧:栈里包含栈帧。

与线程栈相关的内存异常有两个:

a)、StackOverflowError(方法调用层次太深,内存不够新建栈帧)

b)、OutOfMemoryError(线程太多,内存不够新建线程)

1、java.lang.StackOverflowError

栈溢出抛出java.lang.StackOverflowError错误,出现此种情况是因为方法运行的时候,请求新建栈帧时,

栈所剩空间小于战帧所需空间。

例如,通过递归调用方法,不停的产生栈帧,一直把栈空间堆满,直到抛出异常 :

示例代码:

packagecom.jvm;/*** 栈溢出

*@authorfeizi

* @time 2015-1-23上午9:13:11*/

public classSOFTest {public voidstackOverFlowMethod(){

stackOverFlowMethod();

}/*** 通过递归调用方法,不停的产生栈帧,一直把栈空间堆满,直到抛出异常 :

*@paramargs*/

public static voidmain(String[] args) {

SOFTest sof= newSOFTest();

sof.stackOverFlowMethod();

}

}

看看控制台运行的效果:

ac48e1d459a69b65bc0e654956f27a88.png

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值