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规范中并没有规定什么时候加载

 

posted @ 2022-03-20 10:19  showMeTheCodes  阅读(31)  评论(0编辑  收藏  举报