JVM
java virtual Machine
好处:一次编写,到处运行。
自动内存管理,垃圾回收功能。
数组下表越界检查。
多态。

理解底层的底层原理:
整体框架:

程序计数器
作用
用于保存 JVM中 下一条所要执行的指令的地址
特点(每个线程都有自己的程序计数器。)
- 线程私有
- CPU会为每个线程分配时间片,当当前线程的时间片使用完以后,CPU就会去执行另一个线程中的代码。
- 程序计数器是每个线程所私有的,当另一个线程的时间片用完,又返回来执行当前线程的代码时,通过程序计数器可以知道应该执行哪一句指令。
- 不会存在内存溢出
虚拟机栈:线程运行时需要的内存空间。
每个虚拟机栈内是由多个栈帧构成的。
定义
-
每个线程运行需要的内存空间,称为虚拟机栈
-
每个栈由多个栈帧组成,对应着每次调用方法时所占用的内存
-
每个线程只能有一个活动栈帧,对应着当前正在执行的方法
演示
栈帧:每个方法运行时占用的内存。
- 垃圾回收是否涉及栈内存?
- 不需要。因为虚拟机栈中是由一个个栈帧组成的,在方法执行完毕后,对应的栈帧就会被弹出栈。所以无需通过垃圾回收机制去回收内存。
- 栈内存的分配越大越好吗?
- 不是。因为物理内存是一定的,栈内存越大,可以支持更多的递归调用,但是可执行的线程数就会越少。
- 方法内的局部变量是否是线程安全的?
- 如果方法内局部变量没有逃离方法的作用范围,则是线程安全的
- 如果如果局部变量引用了对象,并逃离了方法的作用范围,则需要考虑线程安全问题
Java.lang.stackOverflowError 栈内存溢出
发生原因
- 虚拟机栈中,栈帧过多(无限递归)
- 每个栈帧所占用过大

java.lang.OutofMemoryError :java heap space. 堆内存溢出
堆内存诊断
jps
查看当前系统中有哪些java进程
jmap
查看对内存的占用情况
jmap -heap 进程id
jconsole
图形界面的,多功能检测工具,可以连续检测。
方法区:

内存溢出
- 1.8以前会导致永久代内存溢出
- 1.8以后会导致元空间内存溢出
运行时常量池
- 常量池
- 就是一张表(如上图中的constant pool),虚拟机指令根据这张常量表找到要执行的类名、方法名、参数类型、字面量信息
- 运行时常量池
- 常量池是.class文件中的,当该*类被加载以后,它的常量池信息就会放入运行时常量池,并把里面的符号地址变为真实地址**
常量池与串池的关系

public class Main {
public static void main(String[] args) {
//"a" "b" 被放入串池中,str则存在于堆内存之中
String str = new String("a") + new String("b");
//调用str的intern方法,这时串池中没有"ab",则会将该字符串对象放入到串池中,此时堆内存与串池中的"ab"是同一个对象
String st2 = str.intern();
//给str3赋值,因为此时串池中已有"ab",则直接将串池中的内容返回
String str3 = "ab";
//因为堆内存与串池中的"ab"是同一个对象,所以以下两条语句打印的都为true
System.out.println(str == st2);
System.out.println(str == str3);
}
}
动态拼接的字符串一开始都是在堆内的,不是放在常量池中的。
intern(),就是将字符串对象尝试放入串池,如果有就不会放入。没有的话就会放入。两种都会将串池中的对象返回。

1181

被折叠的 条评论
为什么被折叠?



