一、JVM内存区域

一、Java虚拟机运行时数据区

Java虚拟机运行时数据区,分为五个区。
线程私有程序计数器虚拟机栈本地方法栈
线程共享方法区
java虚拟机运行时数据区

1.程序计数器

如果虚拟机正在执行java方法,则计数器记录的是正在执行的虚拟机字节码指令的地址。执行的是本地方法,计数器值应该为空。

程序计数器可以看作当前线程所执行的字节码的行号指示器,改变这个计数器的值来选取下一条执行的字节码指令。

在程序中,分支、循环、跳转等操作中都依赖这个计数器。

在多线程中,为了线程切换后能恢复到原来位置,线程计数器之间互不影响,所以每个线程的程序计数器是私有的

所以在做多线程的来回切换时,使用if来判断标志进行sleep或者唤醒,会造成虚假唤醒

2.Java虚拟机栈

虚拟机栈主要服务于Java方法

每个java方法被执行时,虚拟机会创建一个栈帧,用于存放局部变量表、操作数栈、动态连接、方法出口等信息。

一个方法被调用到执行完毕,对应着一个栈帧在虚拟机栈入栈到出栈的过程。

局部变量表:存放基本类型(boolean、byte、char、short、int、float、long、double)+对象引用(reference)+returnAddress类型(字节码指令的地址)
局部变量槽:除long和double占用两个变量槽,其他只占用一个变量槽。

OOM异常

  • 栈帧太大:如定义本地变量非常多。
  • 虚拟机栈容量过小
  • 线程过多,请求的栈深度超过了虚拟机所允许的最大深度。

导致新的栈帧内存无法分配时则会抛出异常。

3.本地方法栈

与虚拟机栈类似。本地方法栈为本地方法服务。

4.Java堆

Java堆为对象的分配服务,是运行时数据区最大的一块。“所有的对象实例以及数组都应当在堆上分配”—《Java虚拟机规范》

几乎所有的对象实例分配都在堆上。当然不是所有,像常量、静态变量、被虚拟加载的类型信息在方法区

垃圾收集主要在堆上进行,而目前大部分垃圾收集器都是根据分代收集理论进行的。

JDK1.7及之后版本,字符串常量池也在Java堆中。
String::intern() :是一个本地方法,返回一个字符串常量池中这个字符串引用,没有则添加这个String对象包含的字符串至常量池后返回这个对象的引用。

OOM异常

对象分配过多,并且无法被回收,超出了堆内存。

5.方法区

方法区用于存储被虚拟机加载的类型信息常量静态变量即时编译器编译后的代码缓存等数据。

以前方法区被称为“永久代”,意思就是这些里面的数据不进行回收。当然,用永久代实现方法区,造成内存的溢出只是时间问题。所以在JDK1.8,完全放弃了使用永久代,而是使用在本地内存实现的元空间。

这部分区域的垃圾回收主要是针对常量池的回收类型的卸载

5.1.运行时常量池

运行时常量池是方法区的一部分,Class文件除了有类的版本、字段、方法、接口等描述信息。还有就是运行时常量池。

运行时常量池主要存放编译生成各种字面量与符号引用

直接内存

不属于运行时数据区。JDK1.4引入NIO类,可以通过Native函数在堆外分配内存。

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值