1. java代码是如何运行起来的
(1)Test.java --> javac --> Test.class --> java Test
(2)Test.java --> javac --> Test.class --> Mall.jar --> java -jar Test.jar
(3)Test.java --> javac --> Test.class --> Test.war --> Tomcat --> startup.sh --> org.apache.catalina.startup.Bootstrap
总结起来,运行一个java程序都是通过JDK下面bin/java启动一个JVM虚拟机,在虚拟机里面运行Test.class
1.画一下JVM整个运行原理图
2.JVM内存结构划分
3. JVM中哪些区域是线程私有,哪些是线程共享呢?
(1)线程共享有堆和元空间(方法区),有线程安全问题
(2)其他为私有(虚拟机栈,本地方法栈,程序计数器),没有线程安全问题
4. JVM的程序计数器的特点及作用
- 程序计数器是一块较小的区域,空间可以忽略不计
- 是当前线程执行字节码的行号指示器
- java多线程运行时,每个线程有自己独立的线程程序计数器,他们互不干扰
- 该区域是线程私有,每个线程独立存储
- 不存在OutofMemoryError异常
- 随线程而生,随线程而灭,不需要GC
5. JVM的虚拟机栈的特点及作用
- 线程私有
- 方法执行会创建栈帧,存储局部变量表等信息
- 方法运行,入虚拟机栈,方法执行完,退出虚拟机栈,先入后出。
- 随线程而生,随线程而灭,不需要GC
- 栈的深度大于虚拟机最大设定,报StackOverflowError,比如递归调用过多
public class StackTest { private int stackDeep = 1; public void deepAddOne(){ stackDeep++; deepAddOne(); } public static void main(String[] args) { StackTest stackTest = new StackTest(); try { stackTest.deepAddOne(); }catch (Throwable e){ System.out.println("stackDeep = " + stackTest.stackDeep); throw e; } } }
- 栈需扩展而无法申请空间,报出OutofMemoryError(比较少见),常用的hotspot虚拟机不会有
- 栈里面运行方法,存储方法局部变量,变量指向的的值(常量值,对象值)都存在堆里面
- 栈不设置大小,栈所占用的空间很小,默认只有1M,也可以自己设定。-Xss256K 则是设为256K,比如上面递归了,我调到256K,结果递归从18625次变成了1891次
6. JVM的本地方法栈的特点及作用
和虚拟机栈基本一样
- 存储native方法,
- HotSpot将虚拟机栈和本地方法栈合并了
- 有OutofMemoryError(比较少见)和StackOverflowError
- 随线程而生,随线程而灭,不需要GC
虚拟机栈和本地方法栈和程序计数器随线程而生,随线程而灭,不需要GC
结语
这部分是第三期JVM学习分享,欢迎大家讨论学习分享,你的三连是我最大的动力,第四期会马上出哦,敬请期待