以下文章来自:https://www.jianshu.com/p/8a89fb6d839c,这篇文章涉及多方面知识,所以我在有些地方插入了一些更加深入的文章(方法和函数区别、指针变量、修改引用的值 与 修改引用、函数参数:形参和实参的区别)
JVM数据区
先上一张Java虚拟机运行时数据区中堆、栈以及方法区存储数据的概要图,如下所示:
然后我们来具体解析一下堆和栈
堆
堆是存储时的单位,对于绝大多数应用来说,这块区域是 JVM 所管理的内存中最大的一块。线程共享,主要是存放对象实例和数组。
栈
栈是运行时的单位,Java 虚拟机栈,线程私有,生命周期和线程一致。描述的是 Java 方法执行的内存模型:每个方法在执行时都会创建一个栈帧(Stack Frame)用于存储局部变量表(形参也是局部变量)、操作数栈、动态链接、方法出口等信息。每一个方法从调用直至执行结束,就对应着一个栈帧从虚拟机栈中入栈到出栈的过程。
局部变量表:存放了编译期可知的各种基本类型(boolean、byte、char、short、int、float、long、double)、对象引用(reference 类型)和 returnAddress 类型(指向了一条字节码指令的地址)
堆和栈的对比
一、栈解决程序的运行问题,即程序如何执行,或者说如何处理数据;堆解决的是数据存储的问题,即数据怎么放、放在哪儿。
二、栈因为是运行单位,因此里面存储的信息都是跟当前线程相关的信息。包括:局部变量(含形参)、程序运行状态、方法返回值等等;而堆只负责存储对象信息。
三、在方法中定义的一些基本类型的变量和对象的引用变量都是在函数的栈内存中分配(这一句里面的方法和函数感觉有点不对,方法和函数区别)。堆内存用于存放由new创建的对象和数组。
四、在Java中一个线程就会相应有一个线程栈与之对应,这点保证了程序的并发运行。
而堆则是所有线程共享的,也可以理解为多个线程访问同一个对象,比如多线程去读写同一个对象的值
五、栈内存溢出包括*Error和OutOfMemoryError。*Error:线程请求的栈深度大于虚拟机所允许的深度。OutOfMemoryError:如果虚拟机栈可以动态扩展,而扩展时无法申请到足够的内存;
堆内存溢出是OutOfMemoryError。如果堆中没有内存完成实例分配,并且堆也无法再扩展时,抛出OutOfMemoryError异常。
代码分析
最后,借助网上看到的一个例子帮助对栈内存,堆内存的存储进行理解:
对于以上这段代码,date为局部变量,i,d,m,y都是形参为局部变量,day,month,year为成员变量。下面分析一下代码执行时候的变化:
1. main
方法开始执行:int date = 9;
date:局部变量,基础类型,引用和值都存在栈中。
2. Test test = new Test();
test:为对象引用,存在栈中,对象(new Test())存在堆中。
3. test.change(date);
调用change(int i)方法,i为局部变量,引用和值存在栈中。当方法change执行完成后,i就会从栈中消失。
上图中i→1234,我就想是不是date→1234,所以就去查了一下文章,函数参数:形参和实参的区别,修改引用的值 与 修改引用
4. BirthDate d1= new BirthDate(7,7,1970);
调用BIrthDate类的构造函数生成对象。
d1为对象引用,存在栈中;对象(new BirthDate())存在堆中;
其中d,m,y为局部变量存储在栈中,且它们的类型为基础类型,因此它们的数据也存储在栈中;
day,month,year为BirthDate对象的的成员变量,它们存储在堆中存储的new BirthDate()对象里面(对象存放在堆中等待垃圾回收);
当BirthDate构造方法执行完之后,d,m,y将从栈中消失。
5.main方法执行完之后。
date变量,test,d1引用将从栈中消失;
new Test(),new BirthDate()将等待垃圾回收器进行回收。