对于JVM我们只学习面试需要用的东西,其他的不用了解,因为对于Java工程师来说,日常开发基本用不到JVM的东西,除非你是开发JVM的,但是开发JVM是用C++。
首先我们要知道正是由于JVM才能保证JAVA程序一次编译到处运行。
先来看JVM执行流程:
需要掌握的几个内容:
1.内存区域划分:
2.类加载:将字节码文件加载到内存中的过程进而构建成类对象
典型的面试题:针对下面的代码,给出程序执行结果
class A { public A() { System.out.println("A的构造方法"); } { System.out.println("A的构造代码块"); } static { System.out.println("A的静态代码块"); } } class B extends A { public B() { System.out.println("B的构造方法"); } { System.out.println("B的构造代码块"); } static { System.out.println("B的静态代码块"); } } public class Test extends B{ public static void main(String[] args) { new Test(); new Test(); } }
答案:
A的静态代码块 B的静态代码块 A的构造代码块 A的构造方法 B的构造代码块 B的构造方法 A的构造代码块 A的构造方法 B的构造代码块 B的构造方法
解析:
双亲委派模型:是类加载中的一个环节(比较靠前的环节)
例子一:加载java.lang.String类
例子二:加载自己写的Test类
以上两个例子的查找规则就称为“双亲委派模型”
问题一:问什么JVM要这么设计(设计这么一套查找的规则)?
答:这样能保证即使你自己写的类和标准库中的类,全限定类名重复了也能顺利加载到标准库中的类。
问题二:如果是自定义类,是否要遵守双亲委派模型?
答:可以遵守也可以不遵守,看你的需求。比如Tomcat加载webapp中的类就没有遵守。因为遵守也没有意义(webapp都找不到其他的地方更找不到你要的类)。
总的来说,双亲委派模型只是JVM中的一个小小的规则和细节,只不过名字很高大上,在JVM中类似的规则和细节还有很多。
3.JVM的垃圾回收机制(GC):对不用的内存进行回收
各类语言对内存释放的处理:
垃圾回收机制的优劣:
问题:C++为什么不采用垃圾回收机制?
答:
垃圾回收要干的事情:
找垃圾(判定对象是否死亡)的两种方法:
1)引用计数法:Python采用的方法
引用计数器的问题:
所以即使是Python和PHP使用GC也不仅靠引用计数还依赖其他的配合。
2)可达性分析法:Java采用的方法
可达性分析的优缺点:
释放垃圾的方法:
1)标记清除算法:
2)复制算法:为了解决标记清除的内存碎片问题而产生的
复制算法的问题:
3)标记整理算法:针对复制算法再做出改进
4)分代算法:并不是新的算法,是把上述几种算法结合起来使用
上面说的“找垃圾”和“释放垃圾”只是算法思想,并没有真正实现 ,在JVM中,真正实现垃圾回收的模块是垃圾收集器。
JVM发展到现在,已经有很多的垃圾收集器了。