JVM-内存

程序计数器(Paogram Counter Register)

程序计数器就是当前线程所执行的字节码的行号的指示器,字节码解释器工作就是通过改变程勋计数器的值来获取下一条指令,如循环、分支、异常处理等就是通过程序计数器来实现的;
由于Java多线程执行的实质是分时占有cpu,一条线程执行,其余线程就处于等待状态;所以每条线程搜需要有一个自己的程序呢计数器来记录自己线程执行到的地方,以供重新占有cpu的时候继续执行;
如果当前线程执行的是java方法,程序计数器保存的就是当前执行的字节码指令的地址;如果当前线程执行的是Native方法,这个计数器记录的值为undefined;所以不会出现OutOfMemoryError的情况;

java虚拟机栈

这里写图片描述

Java虚拟机栈是依赖栈数据结构先进后出,后进先出的特性来描述java方法执行顺序情况,每个Java方法执行都会创建一个栈帧,每个栈帧包括该方法的局部变量表(以变量槽的形式存在,通过他可以完成参数值的传递,非静态方法索引为0的变量槽一般储存该方法所属实例的的应用,this关键字通过他实现),操作数栈(在这里对数据进行具体的指令操作),动态链接(与常量池的符号引用变为直接引用有关),方法出口信息(保存了该方法执行完毕需要回到的地址,一般是程序计数器的值);每个栈帧压入虚拟机栈和弹出虚拟机栈就代表了该java方法的开始和完成;
如果线程请求栈的深度超过了虚拟机栈的最大深度,则抛出StackOverFlowError;如果该栈可以动态的扩展,如果栈扩展时无法申请到足够的内存,则会报OutOfMemoryError;
每个线程都拥有自己的虚拟机栈;

本地方法栈

本地方法栈基本和虚拟机方法栈一样,只是本地方法栈是为native方法服务,native方法不限定语言,而虚拟机方法栈只为Java方法服务;

Java堆

Java堆是被虚拟机管理的最大的一块内存,它被所有的线程共享;虚拟机启动时他就会被创建,主要用来存放对象的实例;所以他也是垃圾处理器主要工作的区域;为了便于回收,会将堆分为新生代,老年代;具体来说就是Eden(新的对象被创建的地方),fromSurvivor,toSurvivor。
堆的大小也是可以进行扩展的(可以根据-xmx和-xms惊醒配置)

方法区

方法区和堆一样也是被所有线程共享;由于存放类信息,常量,静态变量,编译器编译后的代码指令数据;对方法区来说,要回收的一般是常量池和类卸载;所以一般来说方法区中的内容不会轻易被回收,所以有时候被称为永久代(存在争议);因为永久代的设置更容易出现内存泄露的情况;

运行时常量池

常量池是方法区的一部分,它保存了编译期间生成的字面量(用于表示固定值的符号)和符号引用(不知道要引用的目标的真实地址,用唯一的一个符号代替);但是在运行期间也是有敞亮进入常量池中的,如string.intern()方法


该文章参考了周志明前辈的《深入理解Java虚拟机》一书,如有侵权,请联系>BlackWyJiang@163.com 删除

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值