目录
–程序计数器(Program Counter Register)(存储当前方法)
–本地方法栈(Native Method Stack)(存储native方法信息)
–方法区(Method Area)(存储类结构/常态/静态变量)
–运行时常量池(Run-Time Constant Pool)(常量运行时表示)
JVM原理图
Java自动内存管理
Java自动内存管理(解决内存泄漏)比如从手动驾驶到自动驾驶
–Java/C#,采用内存自动管理
–程序员只需要申请使用,系统会检查无用的对象并回收内存
–系统统一管理内存,内存使用相对高效,如设置参数不对,也会出现异常
JVM内存-线程私有内存
–程序计数器(Program Counter Register)(存储当前方法)
因为处理器在一个确定是时刻只会执行一个线程中的指令,线程切换后,是通过计数器来记录执行痕迹的,因此程序计数器是每个线程私有的。
程序计数器是一块较小的内存空间,可以看作是当前线程所执行的字节码的行号指示器。分支、循环、跳转、异常处理、线程恢复等基础功能都需要依赖这个计数器来完成。
- Program Counter Register, 一块小内存,每个线程都有
- PC存储当前方法(线程正在执行的方法称为该线程的当前方法)
- 当前方法为本地(native)方法(C语言写的方法)时, pc值未定义(undefined)(计数器的值为空)
- 当前方法为非本地方法(java的方法)时,pc包含了当前正在执行指令的地址(虚拟机字节码指令的地址)
- 当前唯一 一块不会引发OutOfMemoryError异常
–Java虚拟机栈(JVM Stack)(方法的栈帧)
存储栈帧,支持Java方法的调用、执行和退出
- 每个线程有自己独立的Java虚拟机栈,线程私有
- -Xss设置每个线程堆栈大小
- 每个方法从调用到完成对应一个栈帧在栈中入栈、出栈的过程(后进先出(LIFO)栈)
- •栈帧存储局部变量表、操作数栈、动态链接、方法出口等
- •局部变量表存放方法中存在“栈”里面的东西
- 栈的深度超过虚拟机规定深度,StackOverflowError异常
- 无法扩展内存,OutOfMemoryError异常
–本地方法栈(Native Method Stack)(存储native方法信息)
存储native方法(java调用c程序来执行代码,c的方法叫native方法)的执行信息,线程私有
•VM规范没有对本地方法栈做明显规定
•引发的异常:
–栈的深度超过虚拟机规定深度,StackOverflowError异常
–无法扩展内存,OutOfMemoryError异常
JVM内存-多线程共享内存
–堆(Heap) (存储对象和数组)
- 虚拟机启动时创建,所有线程共享, JVM 所管理的最大的一块内存空间(占地最大)
- 对象实例(new操作)和数组都是在堆上分配内存(主要用于存放各种类的实例对象)
- 垃圾回收的主要区域
- 设置大小
- -Xms 初始堆值,-Xmx最大堆值
- 引发的异常
- •无法满足内存分配要求,OutOfMemoryError异常
–方法区(Method Area)(存储类结构/常态/静态变量)
1.又叫静态区,跟堆一样,被所有的线程共享。方法区包含所有的class和static变量。
2.方法区中包含的都是在整个程序中永远唯一的元素,如class,static变量。
通过jvm参数:-XX:PermSize、-XX:MaxPermSize来设置方法区大小
- 存储JVM已经加载类的结构,所有线程共享
- 运行时常量池、类信息、常量、静态变量、即时编译器编译后的代码等
- JVM启动时创建,逻辑上属于堆(Heap)的一部分
- 很少做垃圾回收
- 引发的异常
- 无法满足内存分配要求,OutOfMemoryError异常
- 方法区溢出也是一种常见的内存溢出异常
–运行时常量池(Run-Time Constant Pool)(常量运行时表示)
–Class文件中常量池的运行时表示
–属于方法区的一部分
–动态性
- •Java语言并不要求常量一定只有在编译期产生
- •比如String.intern方法
–引发的异常
- •无法满足内存分配要求,OutOfMemoryError异常
总结