你好JVM,还认识我嘛
你好JVM,我最熟悉的陌生朋友。从刚接触Java的时候我就知道了你,再到后来深入的学习你,可是我发现,每过一段时间我就会把你忘记,一次又一次的。可又不得不重新认识你。于是我决定为了避免我经常把你忘记,把你写进了我的日记,希望能够得到你的允许。
我不止一次看到过这样的你(见图)赤身裸体,曾勾引起我的兴趣,却又渐渐把你忘记。
经过我的苦苦回忆,总算有点印记。记得你不守规矩,到处运行的你却只需一次编译。不管是win、linux还是mac总能看到你的shadow。下面不在扯淡,步入正题!从几个方面来了解JVM
JDK与JVM的区别在哪里
- Jdk是java开发工具包,jvm是java虚拟机。
- jdk中包含了jre「java运行环境」,jre中包含了jvm。
- jdk是整个java的核心,而jvm是java能够实现跨平台运行的关键。
- 编译器把java源文件编译成class文件交给jvm运行。
运行时数据区分为哪几个部分分别代表什么。
运行时数据区分为:方法区,堆,虚拟机栈,本地方法栈以及程序计数器。
- 方法区:用于存放类的信息,常量,静态变量,以及编译后的代码等,它是线程共享的
- 堆:用于存放实例化的数据,同样也线程共享。
- 虚拟机栈:存放局部变量表,操作数据栈等,线程私有。
- 本地方法栈:运行native方法
- 程序计数器:记录当前线程执行位置的计数器。
垃圾回收的几种算法
- 标记/清楚回收算法:在堆中找到正在被引用的对象并进行标记,然后遍历整合堆将未被标记的对象进行垃圾回收。
- 复制算法:在堆中开辟出一段空间1(均分), 实例对象存放在其中一段空间中,在堆空间0中找出正在被引用的对象复制到空间1中,然后清楚空间0的实例对象。
- 标记整理算法:标记出未被引用的对象,然后将有引用的对象向一端移动。
垃圾回收的流程是什么
垃圾回收的流程是先找到垃圾,然后把垃圾回收掉。
找垃圾: 通过引用计数法(已淘汰),对引用进行遍历,找到计数为0的对象实例, 就可以确定这些为可回收对象。但是无法处理循环引用的情况,就会导致内存泄露。
可达性分析:
类似树遍历,查找gc root所有活跃的引用,包括虚拟机栈的栈帧的局部变量表 所引用的对象、本地方法栈的JNI(Java Native Interface)所引用的对象和方法区的静态变量以及常量所引用的对象,把起始点不是gc
root的那些进行回收。找到这些垃圾后,再通过垃圾回收算法把垃圾回收掉(不同区的垃圾回收算法根据其特点会选用不同的垃圾回收算法,后续会专门有笔记来细讲)。
垃圾收集器的分类
串行:serial 是单线程垃圾收集器,执行时会暂停其他线程,采用复制算法,
优点是简单高效。并行:Parallel Scavenge收集器,关注系统和吞吐量,
使用多线程收集。采用复制算法。
JDK1.8中默认使用的是Parallel Scavenge和Parallel Old收集器组合并发:CMS收集器,以获取最短回收停顿时间为目标的收集器。
优点:响应迅速。
堆、栈
堆中存放的都是一个个对象,也就是你在代码中new出来的对象都会存放在这里,那指向这个对象的引用在哪呢?没错,那个引用就存放在栈中。
而堆中又区分年轻代和年老代以及永久代。