JVM——运行时数据区

1 篇文章 0 订阅

JVM 的组成

JVM组成

从上图不难看出,JVM的组成包括下面几个部分:

  • 类加载器(Class Loader)
  • 运行时数据区(Runtime Data Area)
  • 执行引擎(Execution Engine)
  • 本地接口(Native Interface)
  • 本地库(Native Libraries)

首先通过类加载器(Class Loader)把 Java 代码转换成字节码,运行时数据区(Runtime Data Area)再把字节码加载到内存中,而字节码文件只是 JVM 的一套指令集规范,并不能直接交给底层操作系统去执行,因此需要特定的命令解析器执行引擎(Execution Engine),将字节码翻译成底层系统指令,再交由 CPU 去执行,而这个过程中需要调用其他语言的本地接口(Native Interface)与本地库(Native Libraries)来实现整个程序的功能。

运行时数据区

运行时数据区
Java内存区域
运行时数据区包括

  • 程序计数器(Program Counter Register)
  • Java 虚拟机栈(Java Virtual Machine Stacks)
  • 本地方法栈(Native Method Stack)
  • Java 堆(Java Heap)
  • 方法区 (Methed Area)
  • 运行时常量池
  • 直接内存
程序计数器

当前线程所执行的字节码的行号指示器,记录正在执行的虚拟机字节码指令的地址。字节码解析器的工作是通过改变这个计数器的值,来选取下一条需要执行的字节码指令,分支、循环、跳转、异常处理、线程恢复等基础功能,都需要依赖这个计数器来完成。

Java 虚拟机栈

每个 Java 方法在执行的同时会创建一个栈帧用于存储局部变量表、操作数栈、动态链接、方法出口等信息。从方法调用直至执行完成的过程,就对应着一个栈帧在 Java 虚拟机栈中入栈和出栈的过程。栈里面存放着各种基本数据类型和对象的引用。
java虚拟机栈
可以通过 -Xss 这个虚拟机参数来指定每个线程的 Java 虚拟机栈内存大小:

java -Xss512M HackTheJava

该区域可能抛出以下异常

当线程请求的栈深度超过最大值,会抛出 StackOverflowError 异常;

栈进行动态扩展时如果无法申请到足够内存,会抛出 OutOfMemoryError 异常。

写代码使得分别出现StackOverflowError和OutOfMemoryError

分析:堆栈溢出错误一般是递归调用。内存溢出一般是出现在申请了较多的内存空间没有释放。当对象无限被创建且无法被回收时,Java堆区将会出现内存溢出异常

public class Test {
    public static void main(String[] args) {
            test();
    }
    private static void test(){
        test();
    }
}
Exception in thread "main" java.lang.StackOverflowError
	at com.fastdfs.demo.Test.test(Test.java:38)
	at com.fastdfs.demo.Test.test(Test.java:38)
	at com.fastdfs.demo.Test.test(Test.java:38)
	at com.fastdfs.demo.Test.test(Test.java:38)
	at com.fastdfs.demo.Test.test(Test.java:38)
public class Test {
    public static void main(String[] args) {
        List list = new ArrayList();
        for (int i = 0; i < Integer.MAX_VALUE; i++) {
            int[] tmp = new int[1000000];
            list.add(tmp);
        }
    }
}
java.lang.OutOfMemoryError: Java heap space
	at com.fastdfs.demo.Test.main(Test.java:40)
本地方法栈

与虚拟机栈的作用是一样的,只不过虚拟机栈是服务 Java 方法的,而本地方法栈是为虚拟机调用 Native 方法服务的。

Java堆

Java 虚拟机中内存最大的一块,是被所有线程共享的,几乎所有的对象实例都在这里分配内存。 Java堆是垃圾收集的主要区域(“GC 堆”)。

堆不需要连续内存,并且可以动态增加其内存,增加失败会抛出 OutOfMemoryError 异常。

可以通过 -Xms 和 -Xmx 这两个虚拟机参数来指定一个程序的堆内存大小,第一个参数设置初始值,第二个参数设置最大值。

java -Xms1M -Xmx2M HackTheJava
方法区

用于存储已被虚拟机加载的类信息、常量、静态变量、即时编译后的代码等数据。

HotSpot 虚拟机把它当成永久代来进行垃圾回收。但很难确定永久代的大小,因为它受到很多因素影响,并且每次 Full GC 之后永久代的大小都会改变,所以经常会抛出 OutOfMemoryError 异常。为了更容易管理方法区,从 JDK 1.8 开始,移除永久代,并把方法区移至元空间,它位于本地内存中,而不是虚拟机内存中

运行时常量池

运行时常量池是方法区的一部分。

Class 文件中的常量池(编译器生成的字面量和符号引用)会在类加载后被放入这个区域。

除了在编译期生成的常量,还允许动态生成,例如 String 类的 intern()。

直接内存

直接内存
直接内存:不是虚拟机运行时数据区的一部分,也不是java虚拟机规范中定义的内存区域;

如果使用了NIO,这块区域会被频繁使用,在java堆内可以用directByteBuffer对象直接引用并操作;

这块内存不受java堆大小限制,但受本机总内存的限制,可以通过MaxDirectMemorySize来设置(默认与堆内存最大值一样),所以也会出现OOM异常;

深入辨析堆和栈

功能方面: 堆是用来存放对象的,栈是用来执行程序的。
:以栈帧的方式存储方法调用的过程,并存储方法调用过程中基本数据类型的变量(int、short、long、byte、float、double、boolean、char等)以及对象的引用变量,其内存分配在栈上,变量出了作用域就会自动释放。

:堆内存用来存储Java中的对象。无论是成员变量,局部变量,还是类变量,它们指向的对象都存储在堆内存中。

线程独享还是共享
栈内存归属于单个线程,每个线程都会有一个栈内存,其存储的变量只能在其所属线程中可见,即栈内存可以理解成线程的私有内存

堆内存中的对象对所有线程可见。堆内存中的对象可以被所有线程访问。

空间大小
栈的内存要远远小于堆内存,栈的深度是有限制的,可能发生StackOverFlowError问题。

https://mp.weixin.qq.com/s/-fZujwNuj13sZSrHCDzE8Q

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值