JVM调优(一)(二)
一、JVM基础
1、JVM是一种半解释半编译的语言
编译:java文件-->javac编译-->class文件
解释:JVM去解释为二进制文件,交给操作系统去执行
2、JVM、JRE、JDK之间的关系
JRE=JVM+一些java的类库(String.class、Object.class等)
JDK=JRE+自己定义的方法
3、class文件格式:二进制字节流
16进制下是以cafe babe开头的16进制的一堆数字,可以在idea中下载jclasslib插件进行查看
二、类加载器
1、加载过程
①loading:将class文件加载到内存
②linking:
verification:校验class文件是否符合标准,比如class文件开头4字节是否为cafe babe
preparation:将class文件中静态变量赋默认值
resolution:loadClass的第二个参数,loadClass(String name,boolean resolve)
③initializing:静态变量赋初始值
2、java中所有class文件都是由jvm中的类加载器ClassLoader加载到内存中的
class文件加载到内存,内存中有两块空间,一块是class文件的内容,另一个是class文件的类对象,该对象指向class文件的内存
3、类加载器
Bootstrap和Extension是加载指定jar包下的
App:加载自己写的类
4、双亲委派,以自定义类T.class为例(在Custom ClassLoader中加载)
当T.class要被加载到内存中时
①先去Custom ClassLoader中寻找,Custom ClassLoader会在内存中维护一个缓存空间记录该类是否被加载
②若未被加载,向上询问App缓存是否被加载
③若App缓存中未被加载,向上询问Extention中是否被加载
④若Extention缓存中未被加载,向上询问Bootstrap中是否被记载
⑤若Bootstrap缓存中未被加载,向下开始加载
⑥Bootstrap中判断是否能找到该类,发现找不到
⑦到Extention中判断是否能找到该类,发现找不到
⑧到App中判断是否能找到该类,发现找不到
⑨到Custom ClassLoader中判断是否能找到该类,发现可以找到,将T.class记载到内存,如果发现找不到则抛出ClassNotFoundException异常
5、为什么要有双亲委派机制?不能直接加载到内存中吗?
为了安全
如果没有双亲委派机制,自定义一个java.lang.String,直接将该类漏到内存,当用户输入密码的时候,转成自定义的String对象,就可以获取密码
6、父加载器不是类加载器的加载器,也不是类加载器的父类
父加载器只是一个加载的方向
7、可以执行一下下图代码
1 public class T004_ParentAndChild { 2 public static void main(String[] args) { 3 System.out.println("getClassLoader:" + T004_ParentAndChild.class.getClassLoader()); AppClassLoader 4 System.out.println("getClassLoader/getClassLoader:" + T004_ParentAndChild.class.getClassLoader().getClass().getClassLoader()); null 5 System.out.println("getClassLoader.getParent:" + T004_ParentAndChild.class.getClassLoader().getParent()); ExtentionClassLoader 6 System.out.println("getClassLoader.getParent/getParent:" + T004_ParentAndChild.class.getClassLoader().getParent().getParent()); null 7 } 8 }
解释:因为没有自定义加载器,所以第1行执行结果为AppClassLoader
第二行AppClassLoader是加载器是BootstrapClassLoader,而BootstrapClassLoader是由C++实现的,java中没有能与C++直接对应的BootstrapClassLoader,所以显示为null
第三行,获取Parent,就是AppClassLoader的父加载器,为ExtentionClassLoader
第四行,ExtentionClassLoader的父加载器为BootstrapClassLoader,java中没有能与C++直接对应的BootstrapClassLoader,所以显示为null
8、将指定类加载到内存中,直接调用loadClass
1 public class 将指定类加载到内存 { //将test.class通过当前类的classLoader加载到内存中 2 public static void main(String[] args) throws ClassNotFoundException { 3 Class loadClass = 将指定类加载到内存.class.getClassLoader().loadClass("com.test.service.test.class"); 4 System.out.println(loadClass.getName()); 5 } 6 }
使用场景:jreble热部署,将已经loadClass的文件干掉,重新进行loadClass。相当于load了新的class文件
9、自定义类加载器
类加载器的过程:先findInCache在缓存中判断是否被加载,AppClassLoader中查找,如果没有,递归调用父加载器去寻找类,如果最终都没有找到,在findClass中会抛出ClassNotFoundException
那么如果我们需要自定义类加载器的话,只需要继承ClassLoader然后重写findClass方法即可
10、懒加载
JVM规范中并没有规定什么时候加载
作者:http://cnblogs.com/lyc-code/
本文版权归作者和博客园共有,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面明显位置给出原文链接,否则保留追究法律责任的权力。