Java虚拟机在执行Java程序的过程中会把它所管理的内存划分为若干个不同的数据区域,为运行时数据区域,如下图所示:
主要分为五个部分:程序计数器、Java虚拟机栈、本地方法栈、Java堆、方法区。
1、程序计数器
作用:记录下一条需要执行的字节码指令。
特点:线程私有,各条线程之间计数器互不影响,不存在OutOfMemoryError情况
2、Java虚拟机栈
概念:线程运行时候需要的内存空间
虚拟机栈描述的是Java方法执行的线程内存模型:每个方法被执行的时候,Java虚拟机栈都会同步创建一个栈帧用于存储局部变量表、操作数栈、动态连接、方法出口等信息,方法从被调用到执行完毕,就是一次栈帧入栈出栈的过程。
栈帧:每个方法运行时候需要的内存。每个线程只能有一个活动栈帧,对应着当前正在执行的那个方法。
问题辨析:
1)垃圾回收是否会涉及到栈内存? 不会,一般都是回收堆中无用对象
2)栈内存的分配是否越大越好? 不是,栈内存越大反而会使得线程数目变小
3)方法内的局部变量是否是线程安全? 如果方法内局部变量没有逃离方法的作用访问,那么是线程安全的。如果局部变量引用了对象并且逃离了方法的作用范围,需要考虑线程安全问题。
关于栈内存的溢出问题:
1)栈帧过多导致栈内存溢出:StackOverflowError。典型的场景是递归调用没有正确的终止
2)栈帧过大也会导致栈内存溢出
3)如果虚拟机栈容量可以动态扩展,当栈扩展时无法申请到足够的内存会抛出OutOfMemoryError异常。
线程运行诊断:
1)用top定位哪个进程对CPU的占用过高;
2)ps H -eo pid,tid,%cpu | grep进程id(用ps命令进一步定位是哪个线程引起的CPU占用过高);
3)jstack 进程id:可以根据线程id找到有问题的线程,进一步定位到问题代码的源码行号。
3、本地方法栈
本地方法栈与虚拟机栈所发挥的作用非常相似,其区别是本地方法栈是给本地方法运行提供的内存空间,同样是线程私有的,其内存溢出问题和Java虚拟机栈一样。
4、Java堆
此内存区域的唯一目的是存放对象实例,Java世界里“几乎”所有的对象实例都在这里分配内存。
特点:
1)线程共享,需要考虑线程安全问题
2)有垃圾回收机制
3)容易发生OutOfMemoryError:java heap space
堆内存诊断
1)jps工具:查看当前系统中有哪些java进程
2)jmap工具:查看堆内存占用情况 jmap -heap 进程id
3)jconsole工具:图形界面的,多功能监测工具,可以连续监测
5、方法区
1)各个线程共享的内存区域,用于存储已经被虚拟机记载的类型信息、常量、静态变量、即时编译器编译后的代码缓存等数据。
2)如果方法区无法满足新的内存分配需求时,将抛出OutOfMemoryError异常。
运行时常量池是方法区的一部分。常量池就是一张表,虚拟机指令根据这张常量表找到要执行的类名、方法名、参数类型、字面量等信息。常量池是*.class文件中的。而运行时常量池中,当类被加载的时候,常量池的信息就会被放入运行时常量池,并且里面的符号地址变成真实地址。
注意:存在*.class文件中的常量池,在运行期间被JVM装载,并且可以扩充。String的intern()方法就是扩充常量池的一个方法;当一个String实例str调用intern()方法时,Java查找常量池钟思会否会有相同Unicode的字符串常量,如果有,则返回其引用,如果没有,则在常量池中证增加一个Unicode等于str的字符串并且返回它的引用。
StringTable(串池)特性:
1)常量池的字符串仅仅是符号,只有第一次用到的时候才会变成对象
2)利用串池的机制,来避免重复创建字符串对象
3)字符串变量的拼接原理是StringBuilder
4)字符串常量拼接的原理是编译器优化
5)可以使用intern()方法,主动将串池中还没有的字符串对象放入串池