java 虚拟机运行时内存分布分析

  • 虚拟机所管理的内存如下图:

java虚拟机的初始内存和最大内存 java虚拟机占用内存_管理

首先,虚拟机管理内存分为线程共享和线程隔离两大类,线程共享的有方法区和堆;线程隔离数据区有 虚拟机栈、本地方法区和程序计数器。

  • 方法区:这里记录运行时所需的规范类东西,比如类的描述信息,常量,静态变量,即时编译器编译后的代码数据等。这里基本上是不变的,有的虚拟机是用GC分代收集机制来统一管理这里,然后分代的时候分给这里为“永久代”(但这不是一个好方式,更容易内存溢出,所以已经想改变这一策略,1.7hotspot中已经将原本放在这里的字符串常量池移出)。这里存放信息,以供其他地方使用,比如一个对象分配内存后,当真正使用这个对象的时候,需要找到它对应的是哪个类,然后才能知道这个对象里的字节都怎么读取,分别表示什么,否则这个对象里的数据都只是数据而已,这些都是需要对应一个类,这些数据就存在方法区里。这里的内存回收,主要针对常量池和类型卸载。
  • 堆:这里存放对象信息,是java虚拟机所管理的内存中最大的一块。所有的对象和数组都在堆上分配(当然,随着技术的发展,栈上分配,标量替换会导致一些变化,使得这一结论不那么绝对)。堆可以是固定大小的也可以是可扩展的。
  • 虚拟机栈,即通常所说的堆栈中的栈,它是线程私有的,生命周期跟线程相同。这里管理方法执行模型,每一个方法执行时形成一个栈帧,用于存储方法执行所需的局部变量、操作数栈、动态链接、方法出口等。执行完后,栈帧出栈。一个栈帧生成时,它的大小是确定的,在运行时是不会改变局部变量表的大小的。
  • 本地方法栈:这里为本地方法服务,效果跟虚拟机栈一样,hotspot是把这两个区合并使用的。
  • 程序计数器:执行字节码的指针,指向哪个字节码指令,就执行那个指令。如果执行native方法,这里指向空。

以上就是java虚拟机内存管理的一些东西。