JVM内存区域
对于jvm,jre和jdk
–jdk:主要是开发和运行环境
–jre:主要是运行环境
–jvm:java二进制字节码的运行环境,在OS上运行
好处:jvm有自动内存管理机制,可进行垃圾回收
多态,扩展性高
一次编写,多处运行
jvm
内存区域分为线程私有和线程共享和直接内存*
线程共享就是所有线程在内存均可以访问,随虚拟机的启动/关闭而创建/销毁,与java程序运行的生命周期相同
线程私有会在每一个线程中创建,根据线程的启动/结束而创建/销毁
线程共享:方法区,堆
线程私有:虚拟机栈、本地方法栈、程序计数器
能被GC的只有线程共享的部分
内存结构图如下
线程私有
程序计数器
一个记录下一条需要执行指令的字节码,主要功能为:分支、循环、异常处理、线程恢复
不存在内存溢出
虚拟机栈
为java方法服务,Java的基础类型、方法运行的内存空间
一个栈中可有多个栈帧,每个栈帧主要用于存储局部变量表,处理动态链接、异常分派、操作数栈等
调用第一个方法时,会先生成栈,并对该方法生成栈帧后压入栈中,方法执行完毕后出栈,释放内存空间
当调用多个方法时,最底层的会在栈顶,实现先进后出
无论方法是否正常(抛出异常)完成都算方法结束
不涉及GC
方法内的局部变量当没有逃离方法的作用范围时就是线程安全的
本地方法栈
为native方法服务
栈内存溢出
方法递归调用/栈帧多大,当线程请求的栈深度大于虚拟机所允许的深度时
StackOverflowError异常
线程共享
方法区
主要存储被虚拟机加载的类信息、常量、静态变量、即时编译器编译过的代码等数据
在jdk1.8之前会导致永久代内存溢出,1.8+会导致元空间内存溢出
jdk1.6中,方法区内将StringTable放在了常量池中
1.8后将StringTable单独放在堆中,而元空间(本地内存中)有常量池、Class、ClassLoader
ClassLoader 类加载器
加载类的二进制字节码(类基本信息、类常量池、类方法定义)
运行时常量池
当类被加载到虚拟机中,常量池信息会放入到运行时常量池中(此时放入的还只是符号),并把其中的符号地址变为真实地址
存放编译器生成的各种字面量和符号引用,这部分内容在ClassLoader之后存放到运行时常量池中
StringTable 串池,哈希表结构,不能扩容
StringTable存的是常量字符串
只有运行到某行代码,才会将对应的符号–》对象,在创建到串池中,不会提前创建,也不会重复创建
栈
使用new关键字创建的对象都会使用堆内存,且在堆内存的对象需要考虑线程安全的问题
从GC角度还可细分为新生代、老年代
堆内存溢出
OutOfMemoryError异常
不断产生对象并使用时会导致堆内存溢出
可通过设置堆内存值偏小去尽早暴露堆内存溢出的问题