前言
JVM内存结构(也叫内存区域)是JVM中非常重要的部分。JVM内存结构指的是运行时数据区,分为五个部分:方法区、程序计数器、本地方法栈、堆、虚拟机栈。
一、JVM 内存结构
线程私有区域:程序计数器 虚拟机栈 本地方法栈
线程共享区域:堆 方法区
线程私有区域的生命周期与线程相同,随线程的启动而创建,随线程的结束而销毁。
线程共享区域随虚拟机的启动而创建,随虚拟机的关闭而销毁。
二、程序计数器
可以看作是当前线程所执行字节码的行号指示器,指向下一个将要执行的指令代码,由执行引擎来读取下一条指令。是唯一没有内存溢出的区域
一个线程的执行,是通过字节码解释器改变当前线程的计数器的值,来获取下一条需要执行的字节码指令,从而确保线程的正确执行。
三、虚拟机栈和本地方法栈
3.1虚拟机栈
Java 虚拟机栈为 JVM 执行 Java 方法服务,描述的是 Java 方法执行的内存模型。每个方法在执行的同时都会创建一个线帧 (Stack Frame)用于存储局部变量表、操作数栈、动态链接、方法出口等信息,
栈帧用来记录方法的执行过程,在方法被执行时虚拟机会创建一个与之对应的栈帧,方法的执行和返回对于栈帧在虚拟机中的入栈和出栈。
线程1在cpu1上面运行,线程2在cpu2上面运行,在cpu资源不足时其他线程将处于等待状态。在线程内部,每个方法的执行和返回都对应一个栈帧的入栈和出栈,每个运行中的线程当前只有一个栈帧处于活动状态。
3.2本地方法栈
与虚拟机栈的作用是一样的,本地方法栈为 JVM 使用到的 Native 方法服务。
四、堆
在JVM运行过程中创建的对象和产生的数据都被存储在堆中,堆是线程共享的内存区域,也是垃圾收集器进行垃圾回收最主要的内存区域。
Java8 及之后堆内存分为:新生区(新生代)+老年区(老年代)
新生区分为 Eden(伊甸园)区和 Survivor(幸存者)区
栈解决程序的运行问题,即程序如何执行,或者说如何处理数据.
堆解决的是数据存储的问题,即数据怎么放,放在哪儿
五、方法区
方法区同 Java 堆一样是被所有线程共享的区间,用于存储已被虚拟机加载的类信息、常量、静态变量、即时编译器编译后的代码,方法区还包含了一个特殊的区域"运行时常量池"。
方法区看做是一块独立于 java 堆的内存空间
注意:JDK7 及以后的版本中将字符串常量池放到了堆空间中。因为方法区的回收效率很低,我们开发中会有大量的字符串被创建, 回收效率低。放到堆里,能及时回收内存
常量池有什么用 ?
常量池避免了频繁的创建和销毁对象而影响系统性能,其实现了对象的共享。
Integer常量池:Integer 的 valueOf 方法很简单,它判断变量是否在IntegerCache的最小值(-128)和最大值(127)之间,如果在,则返回常量池中的内容,否则 new 一个 Integer 对象。
String 是由 final 修饰的类:
1.String str = new String(“abcd”);
2.String str = “abcd”;
第一种使用 new 创建的对象,存放在堆中。每次调用都会创建一个新的对象。
第二种:先在栈上创建一个 String 类的对象引用变量 str,然后通过符号引用去字符串常量池中找有没有 “abcd”,如果没有,则将“abcd”存放到字符串常量池中,并将栈上的 str 变量引用指向常量池中的“abcd”。如果常量池中已经有“abcd”了,则不会再常量池中创建“abcd”,而是直接将 str 引用指向常量池中的“abcd”。
总结
本文简单的介绍了JVM的内存结构。为了方便记忆和理解,可以从不同区域的功能和特点来记忆。在后续的内容中还将详细介绍堆的分区以及垃圾回收等相关内容供大家参考。