图 1 JVM 体系结构
JVM_ARGS="-Xmx2g -Xms2g -Xmn1024m -XX:PermSize=256m -Xss256k -XX:+DisableExplicitGC -XX:+UseConcMarkSweepGC -XX:+CMSParallelRemarkEnabled -XX:+UseCMSCompactAtFullCollection -XX:LargePageSizeInBytes=128m -XX:+UseFastAccessorMethods -XX:+UseCMSInitiatingOccupancyOnly -XX:+UseCompressedOops -XX:-ReduceInitialCardMarks"
方法区
:存放
JVM
加载的类型信息。包括
:
类型基本信息,常量池,字段信息,方法信息,类变量,指向
ClassLoader
的引用,
Class
类的引用,方法表等。
(对应 JVM 内存配置中的 -PermSize 等)
java 堆 :程序中创建的类的实例和数组,包括 class 对象和 exception 对象,存放在堆里面。堆中除了存储对象的实例数据外,还要存储该对象指向方法区中类型信息的指针。
( JVM 中所有的线程共享堆空间,对应 JVM 内存配置中的 -Xms 和 -Xmx 等)
java 栈 :当 JVM 创建一个新线程时,都会产生线程计数器( PC Register )和栈。每一次方法调用都会产生栈帧,栈帧中包含局部变量区和操作数栈。
( JVM 中栈被线程独享,对应 JVM 内存配置中的 -Xss )
线程计数器 :每个线程拥有自己的程序计数器,它指向下一条指令。当线程调用本地方法的时候, 它为 undefined 。
本地方法栈 :当 JVM 线程调用了本地方法, 则会跳入本地方法栈。本地方法返回后可能再次跳回 java 方法栈。
( JVM 支持本地方法调用,故 JVM 占用的 OS 内存可能会超出 JVM 堆内存大小设置,甚至会产生本地内存泄漏)
附录1: heap
和
stack的区别:
java
的内存分为两类,一类是栈内存,一类是堆内存。栈内存是指程序进入一个方法时,会为这个方法单独分配一块私属存储空间,用于存储这个方法内部的局部变量,当这个方法结束时,分配给这个方法的栈会释放,这个栈中的变量也将随之释放。
堆是与栈作用不同的内存,一般用于存放不放在当前方法栈中的那些数据,例如,使用
new
创建的对象都放在堆里,所以,它不会随方法的结束而消失。方法中的局部变量使用
final
修饰后,放在堆中,而不是栈中。
附录2: 本地方法
如果应用程序需要使用系统特性或设备,比如,调用操作系统的API函数,可能使用Java编写这样的代码是非常麻烦甚至是不可能的。在这种情况下,需要调用其他语言(比如C/C++)编写的代码,这些代码被称为本地(native)方法或本机方法。