多应用+插件架构,代码干净,二开方便,首家独创一键云编译技术,文档视频完善,免费商用码云13.8K 广告
文章出处链接:https://www.zhihu.com/question/19748817/answer/88610988 参考文章:[Jvm系列1—运行时内存结构](http://gityuan.com/2015/10/17/java-memory/) 这个问题,要从线程和内存两个角度来讲解。 **一、线程** 线程从linux角度来看就是task\_struct结构体,是一段可执行的代码,CPU调度的最小单位,这个不是楼主问题的重点,这里就不展开了; **二、其次关于Java栈和的寄存器问题,先来看看jvm的内存模型** jvm内存模型:Java代码是运行在Java虚拟机之上的,由Java虚拟机通过解释执行(解释器)或编译执行(即时编译器)来完成,故Java内存模型,也就是指Java虚拟机的运行时内存模型。 运行时内存模型,分为线程私有和共享数据区两大类,其中线程私有的数据区包含程序计数器、虚拟机栈、本地方法区,所有线程共享的数据区包含Java堆、方法区,在方法区内有一个常量池。java运行时的内存模型图,如下: ![](https://pic4.zhimg.com/50/be87523a50083ebbbd8fa83649568e48_hd.jpg?source=1940ef5c) 从图中,可知内存分为线程私有和共享两大类: (1)线程私有区,包含以下3类: * 程序计数器,记录正在执行的虚拟机字节码的地址; * 虚拟机栈:方法执行的内存区,每个方法执行时会在虚拟机栈中创建栈帧; * 本地方法栈:虚拟机的Native方法执行的内存区; (2)线程共享区,包含以下2类 * Java堆:对象分配内存的区域; * 方法区:存放类信息、常量、静态变量、编译器编译后的代码等数据; * 常量池:存放编译器生成的各种字面量和符号引用,是方法区的一部分。 楼主提到的Java栈,一般而言是指图中的虚拟机栈,在代码中的方法调用过程中,往往需要从一个方法跳转到另一个方法,执行完再返回,那么在跳转之前需要在当前方法的基本信息压入栈中保存再跳转。 **三、关于寄存器的问题** 对于java最常用的虚拟机,sun公司提供的hotspot虚拟机,是基于栈的虚拟机;而对于android的虚拟机,则采用google提供的dalvik,art两种虚拟机,在android 5.0以后便默认采用art虚拟机,这是基于寄存器的虚拟机。 楼主问的是jvm(即java vm),这是基于栈的虚拟机。那么关于虚拟机栈,这块内存的内容,我们再进一步详细分析,如下图: ![](https://pic4.zhimg.com/50/96c5e28ae772a6f0e732cebd39ef57aa_hd.jpg?source=1940ef5c) 可以看到,在虚拟机栈有一帧帧的 栈帧组成,而栈帧包含局部变量表,操作栈等子项,那么线程在运行的时候,代码在运行时,是通过程序计数器不断执行下一条指令。真正指令运算等操作时通过控制操作栈的操作数入栈和出栈,将操作数在局部变量表和操作栈之间转移。