Java 内存模型(一)
打算花比较长的篇幅来描述下自己理解的JVM,尽量描述的清晰易懂一些,从简单慢慢到慢慢深入,一方面自己也复习一下,一方面也供大家参考,少走些弯路。鉴于本人水平有限,如有错误的地方,欢迎指出,感谢。
一段废话引出,大家都知道java有JVM,那JVM有个非常方便的自动内存管理机制,致使java开发人员不再需要为每个new操作去写配对的delete/free代码,而且不容易出现内存泄漏和内存溢出的问题(不容易并非不可能),不过真因为此一旦出现了内存泄漏或者溢出的问题,如果不了解JVM怎么样使用内存的,那么排查起来也会比较困难。
我们初始学习java时候很多人关注就是java的堆栈,这种分法比较粗糙,但是易与理解,实际上远远不止于此,但是今天我们就先按粗浅的方式去解释下,后面我们在逐步深入。按我的习惯,我们先看个例子。
public class BaseTest {
public static void main(String[] args) throws Exception{
ArrayList list=new ArrayList();
while(true){
list.add(new FinalBean());
}
}
}
上面这段代码最终的结果:
java.lang.OutOfMemoryError: GC overhead limit exceeded
Dumping heap to java_pid1288.hprof ...
Heap dump file created [34835053 bytes in 0.247 secs]
Exception in thread "main" java.lang.OutOfMemoryError: GC overhead limit exceeded
at java.util.Arrays.copyOf(Arrays.java:3332)
at java.lang.String.concat(String.java:2032)
at com.cz.smart.center.FinalBean.<init>(FinalBean.java:13)
at com.cz.smart.center.BaseTest.main(BaseTest.java:14)
为什么会出现这种结果呢?
我们通过创建实例对象 new FinalBean(),会开辟堆内存空间,往list列表里面加,导致实例对象没法回收,一直到无法申请到新的内存,导致爆内存溢出,那有没有考虑过,为什么实例对象的内存无法进行回收呢?

再看一个例子
public class BaseTest {
int i = 0;
public static void main(String[] args) throws Exception{
BaseTest b = new BaseTest();
b.addStackLength();
}
public void addStackLength(){
i++;
System.out.println(i);
addStackLength();
}
}
上面这段代码最终的结果:
938
939
Exception in thread "main" java.lang.StackOverflowError
at sun.nio.cs.UTF_8.updatePositions(UTF_8.java:77)
at sun.nio.cs.UTF_8.access$200(UTF_8.java:57)
at sun.nio.cs.UTF_8$Encoder.encodeArrayLoop(UTF_8.java:636)
at sun.nio.cs.UTF_8$Encoder.encodeLoop(UTF_8.java:691)
at java.nio.charset.CharsetEncoder.encode(CharsetEncoder.java:579)
at sun.nio.cs.StreamEncoder.implWrite(StreamEncoder.java:271)
at sun.nio.cs.StreamEncoder.write(StreamEncoder.java:125)
at java.io.OutputStreamWriter.write(OutputStreamWriter.java:207)
at java.io.BufferedWriter.flushBuffer(BufferedWriter.java:129)
at java.io.PrintStream.write(PrintStream.java:526)
at java.io.PrintStream.print(PrintStream.java:597)
at java.io.PrintStream.println(PrintStream.java:736)
为什么这段代码会报这个错呢?StackOverflowError 栈溢出错误。
递归调用,死循环,超过栈的深度,导致抛出异常,那这段代码再内存从怎么跑的呢?我画图解释下:
我们知道栈是线程独享的,每个线程都会有自己独立的栈,而每调用一个方法会产生一个栈针进行入栈。
1.在代码b.addStackLength 时候,会入栈addStackLength方法的栈针,栈针包括局部变量表,操作数栈,动态链接,方法出口等信息(这些后面会详细讲解)。
2.进入方法addStackLength的时候,先执行i++,然后输出i++信息,发现该方法又调用了addStackLength的方法。
3.于是又会入栈一个addStackLength方法的栈针,以此循环,由于是死循环,入栈的栈针超过了栈的深度,抛出StackOverflowError 错误。
注:如果是递归调用而没有死循环,那情况就是在最底层的方法执行完之后,最底层方法就会出栈,调转到上一个调用该方法的下一行,以此类推持续出栈,这就是栈的先入后出。

后续一步步拆分堆和栈内容信息,未完待续。。。
Java 内存模型(一)的更多相关文章
- JVM学习(3)——总结Java内存模型
俗话说,自己写的代码,6个月后也是别人的代码……复习!复习!复习!涉及到的知识点总结如下: 为什么学习Java的内存模式 缓存一致性问题 什么是内存模型 JMM(Java Memory Model)简 ...
- 浅析java内存模型--JMM(Java Memory Model)
在并发编程中,多个线程之间采取什么机制进行通信(信息交换),什么机制进行数据的同步? 在Java语言中,采用的是共享内存模型来实现多线程之间的信息交换和数据同步的. 线程之间通过共享程序公共的状态,通 ...
- JMM(java内存模型)
What is a memory model, anyway? In multiprocessorsystems, processors generally have one or more laye ...
- 《深入理解Java内存模型》读书总结
概要 文章是<深入理解Java内容模型>读书笔记,该书总共包括了3部分的知识. 第1部分,基本概念 包括"并发.同步.主内存.本地内存.重排序.内存屏障.happens befo ...
- Java内存模型深度解析:final--转
原文地址:http://www.codeceo.com/article/java-memory-6.html 与前面介绍的锁和Volatile相比较,对final域的读和写更像是普通的变量访问.对于f ...
- Java内存模型深度解析:volatile--转
原文地址:http://www.codeceo.com/article/java-memory-4.html Volatile的特性 当我们声明共享变量为volatile后,对这个变量的读/写将会很特 ...
- Java内存模型深度解析:顺序一致性--转
原文地址:http://www.codeceo.com/article/java-memory-3.html 数据竞争与顺序一致性保证 当程序未正确同步时,就会存在数据竞争.java内存模型规范对数据 ...
- Java内存模型深度解析:基础部分--转
原文地址:http://www.codeceo.com/article/java-memory-1.html 并发编程模型的分类 在并发编程中,我们需要处理两个关键问题:线程之间如何通信及线程之间如何 ...
- 深入理解java内存模型系列文章
转载关于java内存模型的系列文章,写的非常好. 深入理解java内存模型(一)--基础 深入理解java内存模型(二)--重排序 深入理解java内存模型(三)--顺序一致性 深入理解java内存模 ...
- Java内存模型深度解读
Java内存模型规范了Java虚拟机与计算机内存是如何协同工作的.Java虚拟机是一个完整的计算机的一个模型,因此这个模型自然也包含一个内存模型——又称为Java内存模型. 如果你想设计表现良好的并发 ...
随机推荐
- C++ ORM ODB入门
1.ORM ORM, Object Relational Mapping, 对象关系映射,用来将基于对象的数据结构映射到SQL的数据结构中.即将基于对象的数据映射到关系表中的字段,然后我们可以通过对象 ...
- 机器学习:Jupyter Notebook中numpy的使用
一.Jupyter Notebook的魔法命令 # 模块/方法 + ?或者help(模块/方法):查看模块/方法的解释文档: 1)%run # 机械学习中主要应用两个魔法命令:%run.%timeit ...
- js产生不同的随机数
前言:前几天写到一个程序,用到要使用不同随机数的方法,结果愣是整了半天没整出来,说来也是惭愧啊(亏我还是软件工程的学生,其实这个问题以前遇到过,只是我逃避了,哎,自己刨的坑终究会把自己陷进去,╮(╯▽ ...
- 前端框架:template
ylbtech-前端框架: 1.返回顶部 2.返回顶部 3.返回顶部 4.返回顶部 5.返回顶部 6.返回顶部 作者:ylbtech出处:http://ylbtech.cn ...
- nginx做代理部署WordPress
实验环境:CentOS7 服务器172.16.252.142做Nginx代理服务器: [root@conf.d localhost]#iptables -F [root@conf.d localhos ...
- Ruby中的include
Ruby中的include语句应注意以下两个问题: 1.include与文件无关.C语言中,#include预处理指令在编译期将一个文件的内容插入到另一个文件中.Ruby语句只是简单地产生一个指向指定 ...
- linux普通用户home目录锁定
- hbase java API跟新数据,创建表
package hbaseCURD; import java.io.IOException; import org.apache.hadoop.conf.Configuration; import o ...
- 5种最流行的AI编程语言
人工智能如今正是蓬勃发展的时期,许多开发者都在跃跃欲试,如果你写想转做AI相关的开发,那么来了解更多与AI开发有关的内容吧,本文将介绍创建AI程序时可以使用的5种最佳语言. 人工智能如今正是蓬勃发展的 ...
- 基于OpenCV之视频读取,处理和显示框架的搭建(一)
主要包括以下内容: 1.使用的主要函数的说明. 2.两个实例:视频读取和显示.搭建视频读取和处理框架,调用canny函数提取边缘并显示. 3.一些注意事项和代码说明. 一.使用的主要函数 1.延时函数 ...