JAVA虚拟机的内存

版权声明:有一种生活不去经历不知其中艰辛,有一种艰辛不去体会,不会知道其中快乐,有一种快乐,没有拥有不知其中纯粹 https://blog.csdn.net/wwwzydcom/article/details/84900101

运行时数据区:
在这里插入图片描述

概念模型

线程共享区

方法区

存储运行时常量池,已被虚拟机加载的类信息,常量,静态变量,即时编译器编译后的代码等数据

类信息:类的版本,字段,方法,接口

方法区和永久代

垃圾回收在方法区的行为较少,回收效率低 ,(hotspot省去的方法区)

异常的定义,申请过多也会抛出OutOfMemory

Sun/Oracle JDK的HotSpot VM中,直到JDK7都有“持久代”(Permanent Generation,简称PermGen)。也称为方法区。

Oracle JDK8的HotSpot VM去掉“持久代”,以“元数据区”(Metaspace)替代之。

Oracle是在JDK7的时候宣布移除PermGen内存区域,但是知道JDK8才最终移除。

jdk之前:

  1. 方法区
  2. Internded String

Oracle是在JDK7的时候宣布移除PermGen内存区域,但是知道JDK8才最终移除。

运行时常量
String s1 = "abc";
String s2 = "abc";

System.out.println(s1 == s2); //true

String s3 = new String("abc");

System.out.println( s1 == s3); //false

System.out.println(s1 == s3.intern()); // true

创建abc会在栈内存,局部变量表 s1 和 s2, 堆内存中"abc" s1 和 s2 都指向了同一个"abc"

常量池是在方法区中,会维护一个StringTable 可以想象为一个hashSet

s3 为new,会重新给其分配一个堆内存 所以s1 == s3 比较地址时是false

inter()是将其加入运行时常量,s1 和s2 可以看做是字节常量,运行时常量拿到字节常量的内存,所以是true

直接内存

NIO直接分配堆外内存

java堆

存储对象的实例

java管理最大的区域

新生代,老年代,Eden空间

-Xmx -Xms 修改参数大小

线程独占区

虚拟机栈

(hotpot 不分虚拟机栈和本地方法栈)
存放方法运行时所需的数据,成为栈帧

描述的是java方法执行的动态内存模型

栈帧:每个方法执行,都会创建一个栈帧,伴随着方法从创建到执行完成,用于存储局部变量表,操作数栈,动态链接,方法出口等
局部变量表: 存放编译器可知的各种基本数据类型,引用类型,returnAddress类型
局部变量表的内存空间是在编译期完成分配,当进入一个方法时,这个方法需要在帧分配多少内存是固定的,在方法运行期间是不会改变局部表的大小

StackOverFlowError:栈内存溢出(递归中常见)

public class StackTest {
    private void test(){
        System.out.println("方法在执行");
        test();
    }
    public static void main (String[] args){
            new StackTest().test();
    }
}

本地方法栈

为jvm所调用的native,即本地方法服务

程序计数器

记录当前线程所执行到的字节码的行号

一块较小的内存空间,可以看作是当前线程所执行的字节码的行号指示器
位于线程独占区
线程执行的是java方法,计数器记录的是正在执行的虚拟机字节码指令的地址,如果执行的是native方法,计数器的值为undefined
唯一一个在java虚拟机规范中没有规定任何OutOfMemoryError情况的区域

----> 执行引擎 ----> 本地库接口 ----> 本地方法区

猜你喜欢

转载自blog.csdn.net/wwwzydcom/article/details/84900101