- 虚拟机的基础概念
- class文件结构
- class文件加载过程
- jvm内存模型
- JVM经常使用指令
- GC与调优
JVM加载Class文件主要分3个过程:Loading 、Linking、Initialzingjava
Loading的过程就是经过类加载器将.class
文件加载到jvm内存中过程。须要理解双亲委派机制、类加载器ClassLoader,加载过程以下。
设计模式
不一样的类加载器加载范围不同,以Java8中的为例。tomcat
- BootClassLoader 加载范围
sun.boot.class.paht
- ExtClassLoader 加载范围
java.ext.dirs
- AppClassLoader 加载范围
java.class.path
- CustomClassLoader 可自定义加载范围
前三个加载器来自JDK的Launcher类,三个ClassLoader做为Launcher的内部类,感兴趣能够查看下源码。
开发者也能够自定义的ClassLoader,自定义记载范围。安全
自底向上检查该类是否已经加载,parent方向;自顶向下进行类的实际查找和加载,child方向。
类的加载遵循双亲委派机制,主要是出于安全的考虑。双亲委派机制是如何实现的,下面源码会解释。
注意:双亲委派中存在所谓的父加载器并非加载器的加载器,只是翻译的问题,别混淆了类的继承概念。多线程
ClassLoader源码中比较重要的一个函数是loadClass()
,执行过程是:findLoadedClass()
->parrent.loadClass()
->findClass()
,第一步是自底向上查询是否已经加载,第二步是自顶向下查找加载类。这里就规定或是说实现了双亲委派机制。详细见ClassLoader
的源码。并发
如何自定义ClassLoader?能够继承ClassLoader类,从新本身的findClass()
,在里面调用defineClass()
来实现自定义加载特定范围的类。jvm
从上面的ClassLoader源码中大概能看出是如何实现了双亲委派机制的,从这入手能够经过2种方式打破该机制:函数
- super(parent)指定parent会打破该机制
- 自定义ClassLoader重写
loadClass()
也能够打破
什么时候打破过?双亲委派机制并非不能打破,某些特殊场景下也会选择打破该机制。线程
- JDK 1.2以前,自定义ClassLoader必须重写
loadClass()
,打破过。- 线程ThreadContextClassLoader能够实现基础类调用实现类代码,经过thread.setContextClassLoader指定。
- 热启动热部署,如tomcat都有本身模块指定的classloader,能够加载同一类库的不一样版本。
Class执行方式分为3种:解释执行、编译执行、混合执行,各有优缺点,可经过参数指定。翻译
-Xint
参数指定该模式。-Xcomp
参数指定该模式。-Xmixed
参数指定该模式。热点代码监测:屡次被调用的方法用方法计数器,屡次被调用的循环用循环计数器,可经过参数
-XX:CompileThreshold = 10000
指定触发JIT编译的阈值。
Linking连接的过程分3个阶段:Vertification、Preparation、Resolution。
调用初始化代码clint
,给静态成员变量赋初始值。
这里能够了解下必须初始化的5种状况:
new getstatic putstatic invokestatic
指令,访问final变量除外java.lang.reflect
对类进行反射调用时- 初始化子类的时候,父类必须初始化
- 虚拟机启动时,被执行的主类必须初始化
- 动态语言支持
java.lang.invoke.MethodHandler
解释的结果为REF_getstatic REF_putstatic REF_invokestatic
的方法句柄时,该类必须初始化。
设计模式中单例模式的双重检查的实现,INSTANCE
是否须要加valatile
?
public class Mgr06 { // 是否须要加volatile? private static volatile Mgr06 INSTANCE; private Mgr06() { } public static Mgr06 getInstance() { if (INSTANCE == null) { //双重检查 synchronized (Mgr06.class) { if(INSTANCE == null) { try { Thread.sleep(1); } catch (InterruptedException e) { e.printStackTrace(); } // new 了对象,不为null,但未完成变量的初始化复制,对象处于半初始化状 态,其它线程有可能取到半初始化的对象。 INSTANCE = new Mgr06(); } } } return INSTANCE; } }
我的认为是须要加的。思考方向, class
文件load到内存,给静态变量赋默认值,再赋初始值,new 对象的时候,首先要申请内存空间,而后给成员变量赋默认值,接下来给成员变量赋初始值,这个过程当中对象有可能处于半初始化状态,多线程并发下别的线程有可能取到半初始化的对象,加volatile可保证线程的可见性。
知识分享,转载请注明出处。学无前后,达者为先!