JVM 运行时数据区:
- Java 虚拟机在执行 Java 程序的过程中会把它管理的内存划分为若干个不同的数据区域。
堆 (heap):
- 线程共享
- 存储:绝大部分创建的
实例对象
、数组
、字符串常量池
;- 说明:
– 对象引用存储在栈
,指向存储着对象在堆中的地址
– 堆内存中的对象存储着自己的成员变量,并不保存对象的方法,方法
被保存在帧栈中。- 参数:
-Xms1G
- 初始堆内存-Xmx1G
-最大堆内存;默认大小
– 默认初始值,物理内存的64/1
、最大1/4
;按内存8G来算,就是128M ~ 2G
- 内存回收: 新生代、老年代 -
1:2
; 新生代:eden 、s1 、s2 - 8:1:1
- 回收算法:
– 新生代,复制算法
: 将可用的新生代内存分成2个区域 s1 、s2 每次只使用其中一个区域,对其中一个区域进行回收时,将此区域存活的对象复制到另一个区域,把当前区域清空.eden + s1 - > s2 , eden + s2 ->1
– 老年代,标记压缩
:从根节点标记所有被引用的对象,将所有存活对象压缩移动到内存一端,清除剩下垃圾对象清除。
堆内存溢出(
java.lang.OutOfMemoryError
)
List 循环添加,一般可能出现在:
大批量添加数据
- 所有一般,分批量添加List导入数据库
List<User> list=new ArrayList<>();
while(true){
list.add(new User().setName("小树"));
}
大文件的 IO 处理时:
//size 声明 多 ~~大 ,比如: new byte[in.available()];
byte[] buffer = new byte[size] //1024 * 1024 * 5
异常处理:针对这类型的异常捕捉,用
Throwable e
,然后做出相应处理
try {
//TODO
} catch (Throwable e) {
e.printStackTrace();
}
java 虚拟机栈:
- 线程私有区域,生命周期和线程相同;每个方法被执行的时候,Java虚拟机都会同步创建一个栈帧(Stack Frame)
- 存储:栈帧存储了方法的局部变量表、操作数栈、动态连接和方法返回地址
- 流程:每一个方法被调用直至执行完毕的过程,就对应这一个栈帧在虚拟机栈中从
入栈
到出栈
的过程
–截图取至:地址- 参数:
-Xss<size>
栈溢出 (
java.lang.StackOverflowError
):
- 递归调用方法, 方法调用层次太深,内存不够新建栈帧
private static int count;
public static void count(){
try {
count++;
count();
} catch (Throwable e) {
System.out.println("最大深度:"+count);
e.printStackTrace();
}
}
public static void main(String[] args) {
count();
}
方法区:java类信息、运行时常量池、静态变量
- JDK1.7及以前,称为永久代
Perm
,在堆中
。 存储:java类信息、运行时常量池、静态变量、即时编译器编译后的代码等数据- JDK 1.8,称为
元空间 MetaSpace
,在直接内存中
。 存储:类的信息、常量池放到了本地内存中,将常量池和静态变量放到了Java堆里- 参数:
-XX:MaxMetaspaceSize
程序计数器:保存当前线程锁正在执行的字节码指令的地址;为了线程切换后能恢复到正常执行的位置。
本地方法栈: native方法,JNI,用java 调用 c、c++实现的本地方法库。