JVM 学习
1. JVM 位置
2. JVM体系结构
3. 类加载器
- 类别:
- 虚拟机自带加载器
- 启动类(根)加载器
- 扩展类加载器
- 应用程序加载器
- 类加载器执行次序(双亲委派机制)
- 类加载器收到类加载的请求
- 将这个请求向上委托给父类加载器去完成,一直向上委托,直到启动类加载器(BOOT)
- 启动类加载器检查是否能够加载这个类,能加载就结束,使用当前的加载器,否则,抛出异常,通知子加载器进行加载
- 重复
- 双亲委派机制的作用
- 防止重复加载同一个.class。通过委托去向上面问一问,加载过了,就不用再加载一遍。保证数据安全。
- 保证核心.class不能被篡改。通过委托方式,不会去篡改核心.class,即使篡改也不会去加载,即使加载也不会是同一个.class对象了。不同的加载器加载同一个.class也不是同一个Class对象。这样保证了Class执行安全。
4. Native 、方法区
Native
- native:凡是带了native关键字的,说明java的作用范围达不到了,需要调用c/c++底层的库
- 会进入本地方法栈
- 调用本地方法接口 JNI
- JNI作用:扩展java的使用,融合不同的编程语言为Java所用。最初是为了融合c/c++
- Java诞生的时候c/c++横行,想要立足,必须要有调用c/c++的程序
- 它在内存区域中专门开辟了一块标记区域(Native Method Stack),登记java方法
- 在最终执行的时候,加载本地方法库中的方法通过JNI
- 还可以通过 Socket WebService HTTP等方式调用其他语言
方法区 Method Area
方法区是被所有线程共享,所有字段和方法字节码,以及一些特殊方法,如构造函数、接口代码也再次定义,简单说,所有定义的方法的信息都保存在该区域,此区域属于共享区间
静态变量、常量、类信息(构造方法、接口定义)、运行时的常量池存在方法区中,但是实例变量存在堆内存中,和方法区无关
字符常量池在堆里!
PC寄存器
程序计数器 Program Counter Register
每个线程都有一个程序计数器,是线程私有的,就是一个指针,指向方法区中的方法字节码(用来存储直系那个一条指令的地址,也即姜堰执行的指令代码),在执行引擎读取下一条指令,是一个非常小的内存空间,几乎可以忽略不计。
5. 栈
先进后出
栈:栈内存,主管程序的运行,生病周期和线程同步
线程结束,栈内存也就是释放,对于栈来说,不存在垃圾回收问题
线程一旦结束,栈就over!
栈里面:八大基本类型,对象的引用(地址)、实例的方法
栈运行原理:栈帧
6. 三种jvm
- sun公司的 HotSpot
- BEA
JRockit
- IBM
J9VM
7. 堆 Heap
一个JVM中只有一个堆内存,堆内存的大小是可以调节的。
类加载器读取了类文件后,一般会把什么东西放到堆里? 类,方法,常量,变量~,保存我们所有引用类型的真是对象
堆内存中还要细分为三个区域
- 新生区(伊甸园区)
- 养老区
- 永久区
GC垃圾回收,主要是在伊甸园区和养老区
关于永久区介绍
- 永久区是常驻内存的,存放JDK自身携带的Class对象。Interface元数据,不存在垃圾回收,关闭VM虚拟机会释放这个区域的内存
- 一个启动类,加载了大量的第三方jar包。Tomcat部署了太多的应用,大量动态生成的反射类。不断的被加载。知道内存满,就会出现OOM
- 1.6之前,永久代,常量池是在方法区
- 1.7:永久代,但是慢慢退化了,
去永久代
(符号常量池在堆中,运行时常量池在方法区中) - 1.8 :无永久代,常量池在元空间
- 逻辑上存在,物理上不存在
-Xms : 设置初始化内存分配大小 1/64
-Xmx :设置最大分配内存 1/4
-XX:+PrintGCDetails 打印GC垃圾回收信息
-XX:+HeapDumpOnOutOfMemeryError :OOM dump
GC题目:
- JVM 的内存模型和分区,详细到每个区都放什么
- 对立面的分区有哪些,Eden,from,同,老年区,说说他们的特点
- GC 的算法有哪些,标记清除法,标记压缩(标记整理)法,复制算法,引用计数法,怎么用的?
- 轻GC和重GC分别在什么时候发生
复制算法:
- 好处:没有内存碎片
- 坏处:浪费了内存空间,多了一半空间永远是空(to区)
- 最佳使用场景:对象存活区较低的时候:新生区
标记清除法
- 优点:不需要额外的空间
- 缺点:两次扫描,严重浪费时间,会产生内存碎片
总结
- 内存效率:复制算法>标记清除算法>标记压缩算法
- 内存整齐度:复制算法=标记压缩算法>标记清除算法
- 内存利用率:标记压缩算法=标记清除算法>复制算法
没有最优的算法,只有最合适的算法 --> GC 分代收集算法
-
年轻代:
存活率低,用复制算法 -
老年代:
区域大,存活率高
用标记清除 + 标记压缩混合实现