java的栈
1.当一个新的线程创建时,JVM会为这个线程创建一个新的Stack。一个Java Stack在一个个独立的栈帧中存储了线程的状态。JVM只会在Java Stack中做两个操作:push 和 pop。
2.一个线程当前正在执行的方法称之为线程的 当前方法,当前方法对应的栈帧称为 当前帧,当前方法所属的类称为 当前类,当前类的常量池称为 当前常量池。 在执行一个方法时,JVM会保存当前类和当前常量池的轨迹。当JVM执行 需要操作栈帧中数据的指令时,JVM会在当前栈帧进行处理。
4.栈帧:方法被调用时生成一个栈帧,栈帧由三部分组成:局部变量表、操作数栈以及帧数据。
3.当一个线程执行一个Java方法时,JVM将创建一个新的栈帧并且把它push到栈顶。此时新的栈帧就变成了当前栈帧,方法执行时,使用栈帧来存储参数、局部变量、中间指令以及其他数据。
当方法调用return时,数据将从stack中pop出来,这样程序才能恢复到调用者处继续执行。
递归算法有时会产生非常深的调用从而耗尽栈的空间。
原文链接------作者:提拉没有米苏
一基本了解
java的数据类型分为两种:基本类型和引用类型。基本类型变量保存原始值,引用类型变量保存引用值。引用值代表某个对象的引用,而不是对象本身,对象本身放在这个引用值所表示的地址的位置。
二堆与栈
三说明
栈是运行时的单位,堆是存储的单位
栈解决程序的运行问题,即程序如何执行/如何处理数据。堆解决的是数据的存储问题,即数据放在哪,如何放。
在java中一个线程对应一个线程栈,因不同线程执行逻辑不同,因此线程栈是独立的。而堆是所有线程共享的。栈是运行单位,因此存储的数据与当前线程相关。包括局部变量,程序运行状态,方法返回值等。而堆只负责存储对象信息。
四为什么将堆和栈分开
1.从软件设计的角度分析,栈代表了处理逻辑,堆代表了数据,这样分开,使得处理逻辑更清晰。
2.堆的内容可以被栈共享,是一种有效的数据交互,多个栈可以访问同一个对象。另一方面,堆中共享的常量和缓存可以被栈访问,节省了内存。
3.栈因为运行时需要,比如保存系统运行上下文,需要地址段的划分,栈只能向上增长,因此限制了栈存储内容的能力,而堆是根据需要动态增长,因此栈和堆的拆分使堆动态增长成为可能,相应的栈只需要记住堆中的一个地址即可。
4.面向对象就是堆和栈的完美结合。其实,面向对象方式的程序与以前结构化的程序在执行上没有任何区别。但是,面向对象的引入,使得对待问题的思考方式发生了改变,而更接近于自然方式的思考。当我们把对象拆开,你会发现,对象的属性其实就是数据,存放在堆中;而对象的行为(方法),就是运行逻辑,放在栈中。我们在编写对象的时候,其实即编写了数据结构,也编写的处理数据的逻辑。
在java中,main函数是栈的起始点,也是程序的起始点。
堆中存什么,栈中存什么?
堆中存的是对象,栈中存的是基本数据类型和堆中对象的引用,一个对象的大小不可以估计,或者说是可以动态变化的,但是在栈中,一个对象只对应了一个4byte引用
对象,从某种意义上说,是由基本类型组成的。可以把一个对象看作为一棵树,对象的属性如果还是对象,则还是一颗树(即非叶子节点),基本类型则为树的叶子节点。程序参数传递时,被传递的值本身都是不能进行修改的,但是,如果这个值是一个非叶子节点(即一个对象引用),则可以修改这个节点下面的所有内容。
堆和栈中,栈是程序运行最根本的东西。程序运行可以没有堆,但是不能没有栈。而堆是为栈进行数据存储服务,说白了堆就是一块共享的内存。不过,正是因为堆和栈的分离的思想,才使得Java的垃圾回收成为可能。