Jvm内存结构

栈区

  1. 程序计数器:较小的内存空间, 当前线程执行的字节码的行号指示器;各线程之间独立存储,互不影响;唯一无内存溢出的区域;
  2. java栈:线程私有,它的生命周期与线程相同,它描述的就是java方法执行的内存模型,每个方法在执行的同时都会创建一个栈帧用于存储局部变量表,操作数栈,动态链接,方法出口等信息。方法的执行就对应着栈帧在虚拟机栈中入栈和出栈的过程;栈里面存放着各种基本数据类型和对象的引用;
  3. 本地方法栈:本地方法栈保存的是native方法的信息,当一个JVM创建的线程调用native方法 后,JVM不再为其在虚拟机栈中创建栈帧,JVM只是简单地动态链接并直接调用native方法

堆区

  1. 堆:Java堆是需要重点关注的一块区域,因为涉及到内存的分配(new关键字,反射等)与回收(回收算法,收集器等);
  2. 方法区:也叫永久区,用于存储已经被虚拟机加载的类信息,常量,静态变 量,即时编译器编译后的代码等数据。
  3. 运行时常量池:运行时常量池是方法区的一部分,用于存放编译期生成的各种字面量和符号引用。

直接内存

不是虚拟机运行时数据区的一部分,也不是java虚拟机规范中定义的内存区域;如果使用了NIO,这块区域会被频繁使用,在java堆内可以用directByteBuffer对象直接引用并操作;这块内存不受java堆大小限制,但受本机总内存的限制,可以通过MaxDirectMemorySize来设置(默认与堆内存最大值一样),所以也会出现OOM异常;

堆和栈的区别

  1. 功能 栈是以栈帧的方式存储方法调用的过程,并存储方法调用过程中基本数据类型的变量以及对象的引用变量,其内存分配在栈上,变量出了作用域就会自动释放;

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

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

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

  3. 空间大小:栈的内存要远远小于堆内存;可以通过-Xss选项设置栈内存的大小。-Xms选项设置堆的开始时的大小,-Xmx选项设置堆的最大值

注意:

  1. 永久代来存储类信息、常量、静态变量等数据不是个好主意, 很容易遇到内存溢出的问题,因此在JDK8的实现中将类的元数据放入元空间(直接内存),可以使用MaxMetaspaceSize对元数据区大小进行调整; 将字符串池和类的静态变量放入java堆中.
  2. 对永久代进行调优是很困难的,同时将元空间与堆的垃圾回收进行了隔离,避免永久代引发的Full GC和OOM等问题
Jvm内存结构


分享到:


相關文章: