定义:
Java Virtual Machine,JAVA程序的运行环境(JAVA二进制字节码的运行环境)好处:
1.一次编写,到处运行
2.自动内存管理,垃圾回收机制
3.数组下标越界检查面试题:
JVM JRE JDK的区别(层层的包含关系)
二 内存结构
整体架构
1.程序计数器
作用:记录下一个JVM命令执行地址
特点:
①线程私有:CPU会为每一个线程分配时间片,当当前线程的时间片使用完以后,CPU就会去执行另一个线程的代码,程序计数器是每一个线程所私有的,当另一个线程执行完又返回执行当前线程时,通过程序计数器可以知道一个执行哪一个JVM指令.
②不存在内存溢出
2.虚拟机栈(栈:先进后出)
定义:
每一个线程运行的内存空间,称为虚拟机栈每个栈由多个栈帧(frame)组成,每一个栈帧就是调用方法的内存空间每一个线程只有一个活动栈帧,对应着当前执行的方法
演示
public class Main {
public static void main(String[] args) {
method1();
}
private static void method1() {
method2(1, 2);
}
private static int method2(int a, int b) {
int c = a + b;
return c;
}
}
在控制台中可以看到,主类中的方法在进入虚拟机栈的时候,符合栈的特点
问题:
1.垃圾回收是否涉及栈内存?
不涉及,因为虚拟机栈中由栈帧组成,当方法执行完,对应的栈帧会弹出栈,所以不需要垃圾回收机制去回收内存。
2.栈内存分配越大越好吗?
不是,物理内存是一定的,栈内存越大,可以支持更多的递归调用,但是可执行的线程数就会越少
3.方法内的局部变量是否线程安全?
如果方法的局部变量没有逃离方法的作用范围,则线程安全的
如果局部变量引用了对象,并逃离了方法的作用范围(传入对象的参数,返回对象),则需要考虑线程安全问题
内存溢出
Java.lang.stackOverflowError 栈内存溢出
发生原因
虚拟机栈中,栈帧过多(无限递归)
每个栈帧所占用过大
线程运行诊断
CPU占用过高
Linux环境下运行某些程序的时候,可能导致CPU的占用过高,这时需要定位占用CPU过高的线程
top命令,查看是哪个进程占用CPU过高
ps H -eo pid, tid(线程id), %cpu | grep 刚才通过top查到的进程号 通过ps命令进一步查看是哪个线程占用CPU过高
jstack 进程id 通过查看进程中的线程的nid,刚才通过ps命令看到的tid来对比定位,注意jstack查找出的线程id是16进制的,需要转换
3 本地方法栈
一些带有native关键字的方法就是需要JAVA去调用本地的C或者C++方法,因为JAVA有时候没法直接和操作系统底层交互,所以需要用到本地方法。
4、堆
定义
通过new关键字创建的对象都会被放在堆内存
特点
所有线程共享,堆内存中的对象都需要考虑线程安全问题
有垃圾回收机制
堆内存溢出
java.lang.OutofMemoryError :java heap space. 堆内存溢出
堆内存诊断
jps
jmap
jconsole
jvirsalvm



