JVM学习

JVM学习

本博客是根据解密JVM【黑马程序员出品】教学视频学习时,所做的笔记

一、什么是JVM

定义

Java Virtual Machine,JAVA程序的运行环境(JAVA二进制字节码的运行环境)

好处

  • 一次编写,到处运行
  • 自动内存管理,垃圾回收机制
  • 数组下标越界检查

比较

JVM、JRE、JDK的区别(逐级向上、包含的关系)

img

二、内存结构

整体架构

img

1、程序计数器

1.1. 作用

记住下一条jvm指令的执行地址(如图片中最前面的数字就是地址);硬件方面通过【寄存器】

解释:每条jvm指令会经过 解释器-机器码-CPU这个过程。而程序计数器的作用就是将下一条jvm指令的地址进行保存,解释器就会到程序计数器中读取下一条jvm指令所在的位置。

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-uFEl2Vz4-1669108012071)(C:\Users\vanpersie\AppData\Roaming\Typora\typora-user-images\image-20221111153753188.png)]

1.2. 特点
  • 线程私有
    • CPU会为每个线程分配时间片,当当前线程的时间片使用完以后,CPU就会去执行另一个线程中的代码。当另一个线程的时间片消耗完后,再返回来继续执行当前的线程。
    • 程序计数器是每个线程私有的,当另一个线程的时间片用完,又返回来执行当前线程的代码时,通过程序计数器可以知道应该继续执行哪一句指令
  • 不会存在内存溢出 (在java虚拟机中唯一不会存在内存溢出的区)

2、虚拟机栈

2.1. 定义(先进后出)
  • 每个线程运行需要的内存空间,称为虚拟机栈
  • 每个栈由多个栈帧组成。(栈帧是每个方法运行时所需要的内存)
  • 每个线程只能有一个活动栈帧。(活动栈帧是当前正在执行的方法,同时只能有一个方法在执行,所以每个线程只能有一个活动栈帧)
2.2. 演示

代码

public class Main 
{
	public static void main(String[] args) 
    {
		method1();
	}

	private static void method1() 
    {
		method2(1, 2);
	}

	private static int method2(int a, int b) 
    {
		int c = a + b;
		return c;
	}
}

img

在控制台中可以看到,主类中的方法在进入虚拟机栈的时候,符合栈的特点

2.3. 问题辨析
  • 垃圾回收是否涉及栈内存?

    • 不需要。因为虚拟机栈中是由一个个栈帧组成的,在方法执行完毕后,对应的栈帧就会被弹出栈。所以无需通过垃圾回收机制去回收内存。
  • 栈内存的分配越大越好吗?

    • 不是。因为物理内存是一定的,栈内存越大,可以支持更多的递归调用,但是可执行的线程数就会越少。
  • 方法内的局部变量是否是线程安全的?

    • 如果方法内局部变量没有逃离方法的作用范围,则是线程安全的 (私有的不用考虑线程安全)

      /**
       * 局部变量的线程安全问题
       */
      public class Demo1_18 {
      
          // 多个线程同时执行此方法
          static void m1() {
              int x = 0;
              for (int i = 0; i < 5000; i++) {
                  x++;
              }
              System.out.println(x);
          }
      }
      
    • 如果局部变量引用了对象,并逃离了方法的作用范围,则需要考虑线程安全问题 (共享的需要考虑线程安全)

      /**
       * 局部变量的线程安全问题
       */
      public class Demo1_17 {
          public static void main(String[] args) {
              StringBuilder sb = new StringBuilder();
              sb.append(4);
              sb.append(5);
              sb.append(6);
              new Thread(()->{
                  m2(sb);
              }).start();
          }
      
          public static void m1() { //不会有线程安全的问题;因为是局部变量,属于私有的
              StringBuilder sb = new StringBuilder();
              sb.append(1);
              sb.append(2);
              sb.append(3);
              System.out.println(sb.toString());
          }
      
          public static void m2(StringBuilder sb) { //会有线程安全问题;其它的线程也能访问到它,它不再是线程私有的,是共享的
              sb.append(1); //局部变量引用了对象
              sb.append(2);
              sb.append(3);
              System.out.println(sb.toString());
          }
      
          public static StringBuilder m3() { //会有线程安全问题;由于我们把结果返回了。其它的线程会拿到返回结果,所以它不再是私有的
              StringBuilder sb = new StringBuilder();
              sb.append(1);
              sb.append(2);
              sb.append(3);
              return sb; //逃离了方法的作用范围
          }
      }
      

2.4. 内存溢出

Java.lang.stackOverflowError 栈内存溢出

发生原因

  • 每个栈帧所占用过大

  • 虚拟机栈中,栈帧过多(无限递归)

无限递归案例一:

public class Demo2
{
    private static int count;
    
    public static void main(String[] args)
    {
        try
        {
            method1();
        }
        catch (Throwable e)
        {
            e.printStackTrace();
            System.out.println(count);
        }
    }

    private static void method1()
    {
        count++;
        method1();
    }
}

出现了栈溢出的错误:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-Cv75eVIr-1669108012071)(C:\Users\vanpersie\AppData\Roaming\Typora\typora-user-images\image-20221111171350830.png)]

无限递归案例二:

产生原因:打印时部门会调用emps这个集合,集合中的每个元素又会调用Emp这个类,Emp中又包含Dept这个属性,会导致无限递归的产生进而导致栈溢出。所以为了避免这种现象,可以使用@JsonIgnore注解,忽略这个属性不将这个属性转化为Json。

/*json 数据转换*/
public class Demo3
{
    public static void main(String[] args) throws JsonProcessingException
    {
        Dept d = new Dept();
        d.setName("Market");

        Emp e1 = new Emp();
        e1.setName("zhang");
        e1.setDept(d);

        Emp e2 = new Emp();
        e2.setName("li");
        e2.setDept(d);

        d.setEmps(Arrays.asList(e1, e2));

        // { name: 'Market', emps: [{ name:'zhang', dept:{ name:'', emps: [ {}]} },] }
        ObjectMapper mapper = new ObjectMapper();
        System.out.println(mapper.writeValueAsString(d));
    }
}

class Emp
{
    private String name;
    //@JsonIgnore //转化为Json时忽略这个属性
    private Dept dept;

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public Dept getDept() {
        return dept;
    }

    public void setDept(Dept dept) {
        this.dept = dept;
    }
}

class Dept
{
    private String name;
    private List<Emp> emps;

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public List<Emp> getEmps() {
        return emps;
    }

    public void setEmps(List<Emp> emps) {
        this.emps = emps;
    }
}

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-dnB7EWrZ-1669108012072)(C:\Users\vanpersie\AppData\Roaming\Typora\typora-user-images\image-20221111172458996.png)]

2.5. 线程运行诊断

案例一: CPU占用过高

  • Linux环境下运行某些程序的时候,可能导致CPU的占用过高,这时需要定位占用CPU过高的线程

    • 用top定位哪个进程对cpu的占用过高

      [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-kxJJvn74-1669108012072)(C:\Users\vanpersie\AppData\Roaming\Typora\typora-user-images\image-20221111180920652.png)]

    • ps H -eo pid, tid(线程id), %cpu | grep 刚才通过top查到的进程号 通过ps命令进一步查看是哪个线程占用CPU过高

      [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-1Qmxev1j-1669108012072)(C:\Users\vanpersie\AppData\Roaming\Typora\typora-user-images\image-20221111180948217.png)]

    • jstack + 进程id 通过查看进程中的线程的nid,刚才通过ps命令看到的tid来对比定位,注意jstack查找出的线程id是16进制的需要转换

      [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-ZFmkRxwi-1669108012073)(C:\Users\vanpersie\AppData\Roaming\Typora\typora-user-images\image-20221111181000979.png)]

通过上述方式找到了源代码CPU消耗过高的文件及行号

​ [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-pRClujcN-1669108012073)(C:\Users\vanpersie\AppData\Roaming\Typora\typora-user-images\image-20221111181021921.png)]

案例二:程序运行很长时间没有结果,如何诊断案例

jstack + 进程Id

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-9E9PBWSd-1669108012073)(C:\Users\vanpersie\AppData\Roaming\Typora\typora-user-images\image-20221111181720323.png)]

/**
 * 演示线程死锁
 */
class A{};
class B{};
public class Demo1_3 {
    static A a = new A();
    static B b = new B();


    public static void main(String[] args) throws InterruptedException {
        new Thread(()->{
            synchronized (a) {
                try {
                    Thread.sleep(2000);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
                synchronized (b) { //第21行出现死锁
                    System.out.println("我获得了 a 和 b");
                }
            }
        }).start();
        Thread.sleep(1000);
        new Thread(()->{
            synchronized (b) {
                synchronized (a) { //第29行出现死锁
                    System.out.println("我获得了 a 和 b");
                }
            }
        }).start();
    }
}

3、本地方法栈

一些带有native关键字的方法就是需要JAVA去调用本地的C或者C++方法,因为JAVA有时候没法直接和操作系统底层交互,所以需要用到本地方法(C或C++)

4、堆

4.1. 定义

通过new关键字创建的对象都会被放在堆内存

4.2. 特点
  • 所有线程共享,堆内存中的对象都需要考虑线程安全问题
  • 有垃圾回收机制
4.3. 堆内存溢出
/**
 * 演示堆内存溢出 java.lang.OutOfMemoryError: Java heap space
 * -Xmx8m:可以调整堆的空间大小,改为8mb
 */
public class Demo1_5 
{
    public static void main(String[] args) 
    {
        int i = 0;
        try 
        {
            List<String> list = new ArrayList<>();
            String a = "hello";
            while (true) 
            {
                list.add(a); // hello, hellohello, hellohellohellohello ...
                a = a + a;  // hellohellohellohello
                i++;
            }
        } 
        catch (Throwable e) 
        {
            e.printStackTrace();
            System.out.println(i);
        }
    }
}

生产环境建议:如果内存比较大,内存溢出不会那么快的暴露;这时,我们可以将堆内存调小,让内存溢出尽早暴露

4.4. 堆内存诊断
  • **jps工具:**查看当前系统中有哪些java进程

    [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-dzD7ZAiT-1669108012073)(C:\Users\vanpersie\AppData\Roaming\Typora\typora-user-images\image-20221111185252164.png)]

  • jmap工具:查看堆内存占用情况 jmap -heap pid

    [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-e3CC6n1u-1669108012074)(C:\Users\vanpersie\AppData\Roaming\Typora\typora-user-images\image-20221111185312414.png)]

  • jstack 工具:线程监控

  • **jconsole工具:**图形界面的,多功能的检测工具,可以连续监测

    [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-VbpTEEGv-1669108012074)(C:\Users\vanpersie\AppData\Roaming\Typora\typora-user-images\image-20221111185651933.png)]

    [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-ZWoGAaXo-1669108012074)(C:\Users\vanpersie\AppData\Roaming\Typora\typora-user-images\image-20221111185724359.png)]

    然后选择不安全连接即可观测

  • **jvisualvm工具:**图形界面的,多功能的检测工具,可以连续监测;还有dump

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-kQyIJLCI-1669108012075)(C:\Users\vanpersie\AppData\Roaming\Typora\typora-user-images\image-20221111191039477.png)]

案例一:jps、jmap、jconsole的使用演示
/**
 * 演示堆内存
 */
public class Demo1_4 
{
    public static void main(String[] args) throws InterruptedException 
    {
        System.out.println("1...");
        Thread.sleep(30000);
        byte[] array = new byte[1024 * 1024 * 10]; // 10 Mb
        System.out.println("2...");
        Thread.sleep(20000);
        array = null;
        System.gc(); //进行垃圾回收
        System.out.println("3...");
        Thread.sleep(1000000L);
    }
}

打印1… 后Eden中使用的空间

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-JWdidsAO-1669108012075)(C:\Users\vanpersie\AppData\Roaming\Typora\typora-user-images\image-20221111185334521.png)]

打印2… 后Eden中使用的空间;used的内存明显增加

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-fDhZfK8r-1669108012075)(C:\Users\vanpersie\AppData\Roaming\Typora\typora-user-images\image-20221111185351024.png)]

打印3… 后Eden中使用的空间;used的内存明显减少

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-BgPWPBo1-1669108012075)(C:\Users\vanpersie\AppData\Roaming\Typora\typora-user-images\image-20221111185402283.png)]

案例二: 垃圾回收后,内存占用仍然很高,排查方式案例(jvisualvm使用的演示)
/**
 * 演示查看对象个数 堆转储 dump
 */
public class Demo1_13 
{
    public static void main(String[] args) throws InterruptedException 
    {
        List<Student> students = new ArrayList<>();
        for (int i = 0; i < 200; i++) 
        {
            students.add(new Student());
        }
        Thread.sleep(1000000000L);
    }
}

class Student 
{
    private byte[] big = new byte[1024*1024];
}

解决方式:jvisualvm 可以使用dump,查找最大的对象堆转储 dump(基于上述问题,使用工具进行查看); 在测试环境下,我们可以开启dump文件记录,然后将dump文件导入到jvisualvm工具查看,占用最多的内存的对象是哪些。

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-RNoZ15VT-1669108012076)(C:\Users\vanpersie\AppData\Roaming\Typora\typora-user-images\image-20221111190459571.png)]

点击Demo5查看

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-90hspRMO-1669108012076)(C:\Users\vanpersie\AppData\Roaming\Typora\typora-user-images\image-20221111190529374.png)]

点击堆dump后再点击查找即可找到占用内存比较多的类

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-nDsYXuS6-1669108012077)(C:\Users\vanpersie\AppData\Roaming\Typora\typora-user-images\image-20221111190725492.png)]

点击ArrayList即可详细观察这个类的情况。

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-5Vjk09j9-1669108012077)(C:\Users\vanpersie\AppData\Roaming\Typora\typora-user-images\image-20221111191802977.png)]

5、方法区

定义

  • 线程共享
  • 在JVM启动时创建,在逻辑上属于堆的一部分(看厂商实现)
  • 方法区也可能会内存溢出
5.1. 结构

img

5.2. 内存溢出
  • 1.8以前会导致永久代内存溢出 (用堆的一部分空间作为方法区)
  • 1.8以后会导致元空间内存溢出(用本地的一部分空间作为方法区)
/**
 * 演示元空间内存溢出 java.lang.OutOfMemoryError: Metaspace
 * -XX:MaxMetaspaceSize=32m : 需要调整内存大小才能展示错误的效果
 */
public class Demo6 extends ClassLoader //类加载器:可以用来加载类的二进制字节码
{
    public static void main(String[] args)
    {
        int j = 0;
        try
        {
            Demo6 test = new Demo6();
            for (int i = 0; i < 100000; i++, j++)
            {
                // ClassWriter 作用是生成类的二进制字节码
                ClassWriter cw = new ClassWriter(0);
                //参数顺序如下:版本号, public, 类名, 包名, 父类, 接口
                cw.visit(Opcodes.V1_8, Opcodes.ACC_PUBLIC, "Class" + i, null, "java/lang/Object", null);
                // 返回 byte[]
                byte[] code = cw.toByteArray();
                // 执行了类的加载
                test.defineClass("Class" + i, code, 0, code.length); // Class 对象
            }
        }
        finally
        {
            System.out.println(j);
        }
    }
}

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-WsXfJN1e-1669108012078)(C:\Users\vanpersie\AppData\Roaming\Typora\typora-user-images\image-20221113102954063.png)]

5.3. 常量池

二进制字节码的组成:类的基本信息、常量池、类的方法定义(包含了虚拟机指令)

通过反编译来查看类的信息

// 二进制字节码(类基本信息,常量池,类方法定义,包含了虚拟机指令)
public class Demo7
{
    public static void main(String[] args)
    {
        System.out.println("hello world");
    }
}

在idea的终端上输入

D:\Ajvm\out\production\jvm\day1>javap -v Demo7.class

反编译效果如下:

D:\Ajvm\out\production\jvm\day1>javap -v Demo7.class
Classfile /D:/Ajvm/out/production/jvm/day1/Demo7.class
  Last modified 2022-11-13; size 528 bytes
  MD5 checksum bbc12c1b031dc3e2a424cbb7056d97d6
  Compiled from "Demo7.java"
public class day1.Demo7
  minor version: 0
  major version: 52
  flags: ACC_PUBLIC, ACC_SUPER     
//以上是类基本信息
      
Constant pool:
   #1 = Methodref          #6.#20         // java/lang/Object."<init>":()V
   #2 = Fieldref           #21.#22        // java/lang/System.out:Ljava/io/PrintStream;
   #3 = String             #23            // hello world
   #4 = Methodref          #24.#25        // java/io/PrintStream.println:(Ljava/lang/String;)V
   #5 = Class              #26            // day1/Demo7
   #6 = Class              #27            // java/lang/Object
   #7 = Utf8               <init>
   #8 = Utf8               ()V
   #9 = Utf8               Code
  #10 = Utf8               LineNumberTable
  #11 = Utf8               LocalVariableTable
  #12 = Utf8               this
  #13 = Utf8               Lday1/Demo7;
  #14 = Utf8               main
  #15 = Utf8               ([Ljava/lang/String;)V
  #16 = Utf8               args
  #17 = Utf8               [Ljava/lang/String;
  #18 = Utf8               SourceFile
  #19 = Utf8               Demo7.java
  #20 = NameAndType        #7:#8          // "<init>":()V
  #21 = Class              #28            // java/lang/System
  #22 = NameAndType        #29:#30        // out:Ljava/io/PrintStream;
  #23 = Utf8               hello world
  #24 = Class              #31            // java/io/PrintStream
  #25 = NameAndType        #32:#33        // println:(Ljava/lang/String;)V
  #26 = Utf8               day1/Demo7
  #27 = Utf8               java/lang/Object
  #28 = Utf8               java/lang/System
  #29 = Utf8               out
  #30 = Utf8               Ljava/io/PrintStream;
  #31 = Utf8               java/io/PrintStream
  #32 = Utf8               println
  #33 = Utf8               (Ljava/lang/String;)V
//以上是常量池信息   
                            
{
  public day1.Demo7();
    descriptor: ()V
    flags: ACC_PUBLIC
    Code:
      stack=1, locals=1, args_size=1
         0: aload_0
         1: invokespecial #1                  // Method java/lang/Object."<init>":()V
         4: return
      LineNumberTable:
        line 3: 0
      LocalVariableTable:
        Start  Length  Slot  Name   Signature
            0       5     0  this   Lday1/Demo7;

  public static void main(java.lang.String[]);
    descriptor: ([Ljava/lang/String;)V
    flags: ACC_PUBLIC, ACC_STATIC
    Code:
      stack=2, locals=1, args_size=1
        //我们根据这些#常量符号去常量池中寻找对应的指令
         0: getstatic     #2                  // Field java/lang/System.out:Ljava/io/PrintStream;
         3: ldc           #3                  // String hello world
         5: invokevirtual #4                  // Method java/io/PrintStream.println:(Ljava/lang/String;)V
         8: return
      LineNumberTable:
        line 7: 0
        line 8: 8
      LocalVariableTable:
        Start  Length  Slot  Name   Signature
            0       9     0  args   [Ljava/lang/String;
}
//以上是类方法定义的信息                                     
SourceFile: "Demo7.java"

5.4. 运行时常量池
  • 常量池
    • 就是一张表(如上图中的constant pool),虚拟机指令根据这张常量表找到要执行的类名、方法名、参数类型、字面量信息
  • 运行时常量池
    • 常量池是*.class文件中的,当该类被加载以后,它的常量池信息就会放入运行时常量池,并把里面的符号地址变为真实地址
5.5. 常量池与串池的关系
串池StringTable

特征

  • 常量池中的字符串仅是符号,只有在被用到时才会转化为对象
  • 利用串池的机制,来避免重复创建字符串对象
  • 字符串变量拼接的原理是StringBuilder
  • 字符串常量拼接的原理是编译器优化
  • 可以使用intern方法,主动将串池中还没有的字符串对象放入串池中
  • 注意:无论是串池还是堆里面的字符串,都是对象

用来放字符串对象且里面的元素不重复

public class StringTableStudy 
{
	public static void main(String[] args) 
    {
		String a = "a"; 
		String b = "b";
		String ab = "ab";
	}
}

常量池中的信息,都会被加载到运行时常量池中,但这是a b ab 仅是常量池中的符号,还没有成为java字符串

0: ldc           #2                  // String a
2: astore_1
3: ldc           #3                  // String b
5: astore_2
6: ldc           #4                  // String ab
8: astore_3
9: returnCopy

当执行到 ldc #2 时,会把符号 a 变为 “a” 字符串对象,并放入串池中(hashtable结构 不可扩容)

当执行到 ldc #3 时,会把符号 b 变为 “b” 字符串对象,并放入串池中

当执行到 ldc #4 时,会把符号 ab 变为 “ab” 字符串对象,并放入串池中

最终StringTable [“a”, “b”, “ab”]

注意:字符串对象的创建都是懒惰的,只有当运行到那一行字符串且在串池中不存在的时候(如 ldc #2)时,该字符串才会被创建并放入串池中。

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-MP7MBEBC-1669108012078)(C:\Users\vanpersie\AppData\Roaming\Typora\typora-user-images\image-20221113115234805.png)]

使用拼接字符串变量对象创建字符串的过程

public class StringTableStudy 
{
	public static void main(String[] args) 
    {
		String a = "a";
		String b = "b";
		String ab = "ab";
		//拼接字符串对象来创建新的字符串
		String ab2 = a+b; 
	}
}

反编译后的结果

	 Code:
      stack=2, locals=5, args_size=1
         0: ldc           #2                  // String a
         2: astore_1
         3: ldc           #3                  // String b
         5: astore_2
         6: ldc           #4                  // String ab
         8: astore_3
         9: new           #5                  // class java/lang/StringBuilder
        12: dup
        13: invokespecial #6                  // Method java/lang/StringBuilder."<init>":()V
        16: aload_1
        17: invokevirtual #7                  // Method java/lang/StringBuilder.append:(Ljava/lang/String;)Ljava/lang/StringBuillder;
        20: aload_2
        21: invokevirtual #7                  // Method java/lang/StringBuilder.append:(Ljava/lang/String;)Ljava/lang/StringBuilder;
        24: invokevirtual #8                  // Method java/lang/StringBuilder.toString:()Ljava/lang/String;
        27: astore        4
        29: returnCopy

通过拼接的方式来创建字符串的过程是:StringBuilder().append(“a”).append(“b”).toString()

最后的toString方法的返回值是一个新的字符串,字符串的和拼接的字符串一致,但是是两个不同的字符串,一个存在于串池之中,一个存在于堆内存之中

String ab = "ab";
String ab2 = a+b;
//结果为false,因为ab是存在于串池之中,ab2是由StringBuffer的toString方法所返回的一个对象,存在于堆内存之中
System.out.println(ab == ab2); //false

使用拼接字符串常量对象的方法创建字符串

public class StringTableStudy 
{
	public static void main(String[] args) 
    {
		String a = "a";
		String b = "b";
		String ab = "ab";
		String ab2 = a+b;
		//使用拼接字符串的方法创建字符串
		String ab3 = "a" + "b";
	}
}

反编译后的结果

 	  Code:
      stack=2, locals=6, args_size=1
         0: ldc           #2                  // String a
         2: astore_1
         3: ldc           #3                  // String b
         5: astore_2
         6: ldc           #4                  // String ab
         8: astore_3
         9: new           #5                  // class java/lang/StringBuilder
        12: dup
        13: invokespecial #6                  // Method java/lang/StringBuilder."<init>":()V
        16: aload_1
        17: invokevirtual #7                  // Method java/lang/StringBuilder.append:(Ljava/lang/String;)Ljava/lang/StringBuilder;
        20: aload_2
        21: invokevirtual #7                  // Method java/lang/StringBuilder.append:(Ljava/lang/String;)Ljava/lang/StringBuilder;
        24: invokevirtual #8                  // Method java/lang/StringBuilder.toString:()Ljava/lang/String;
        27: astore        4
        //ab3初始化时直接从串池中获取字符串
        29: ldc           #4                  // String ab
        31: astore        5
        33: returnCopy
  • 使用拼接字符串常量的方法来创建新的字符串时,因为内容是常量,javac在编译期会进行优化,结果已在编译期确定为ab,而创建ab的时候已经在串池中放入了“ab”,所以ab3直接从串池中获取值,所以进行的操作和 ab = “ab” 一致。
  • 使用拼接字符串变量的方法来创建新的字符串时,因为内容是变量,只能在运行期确定它的值,所以需要使用StringBuilder来创建
intern方法 (针对JDK1.8之后的)

调用字符串对象的intern方法,会将该字符串对象尝试放入到串池中

  • 如果串池中没有该字符串对象,则放入成功
  • 如果有该字符串对象,则放入失败

无论放入是否成功,都会返回串池中的字符串对象

注意:此时如果调用intern方法成功,堆内存与串池中的字符串对象是同一个对象;如果失败,则不是同一个对象

例1

public class Main 
{
	public static void main(String[] args) 
    {
		//"a" "b" 被放入串池中,str则存在于堆内存之中
		String str = new String("a") + new String("b");
		//调用str的intern方法,intern方法会尝试将str放入串池中。这时串池中没有"ab",会将该字符串对象放入到串池中,此时堆内存与串池中的"ab"是同一个对象
		String st2 = str.intern();
		//给str3赋值,因为此时串池中已有"ab",则直接将串池中的内容返回
		String str3 = "ab";
		//因为堆内存与串池中的"ab"是同一个对象,所以以下两条语句打印的都为true
		System.out.println(str == st2);
		System.out.println(str == str3);
	}
}

例2

public class Main 
{
	public static void main(String[] args) 
    {
        //此处创建字符串对象"ab",因为串池中还没有"ab",所以将其放入串池中
		String str3 = "ab";
        //"a" "b" 被放入串池中,str则存在于堆内存之中
		String str = new String("a") + new String("b");
        //此时因为在创建str3时,"ab"已存在与串池中,所以放入失败,但是会返回串池中的"ab"
		String str2 = str.intern();
        //false
		System.out.println(str == str2);
        //false
		System.out.println(str == str3);
        //true
		System.out.println(str2 == str3);
	}
}
intern方法(针对JDK1.6的)

调用字符串对象的intern方法,会将该字符串对象尝试放入到串池中

  • 如果串池中没有该字符串对象,会将该字符串对象复制一份,再放入到串池中
  • 如果有该字符串对象,则放入失败

无论放入是否成功,都会返回串池中的字符串对象

注意:此时无论调用intern方法成功与否,串池中的字符串对象和堆内存中的字符串对象都不是同一个对象

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-NiEmWk9G-1669108012079)(C:\Users\vanpersie\AppData\Roaming\Typora\typora-user-images\image-20221113122751367.png)]

5.6. StringTable位置

JDK1.6版本,StringTable存放在常量池中;
JDK1.7 及之后版本的 JVM 将 StringTable存放到了堆中;img

JDK1.8 字符串常量池在堆中实例验证

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-9InXXBHL-1669108012079)(C:\Users\vanpersie\AppData\Roaming\Typora\typora-user-images\image-20221113124814319.png)]

img

5.7. StringTable 垃圾回收

StringTable在内存紧张时,会发生垃圾回收

/**
 * 演示 StringTable 垃圾回收
 * -Xmx10m : 设置虚拟机堆内存的最大值
 * -XX:+PrintStringTableStatistics :字符串表的统计信息
 * -XX:+PrintGCDetails -verbose:gc :打印垃圾回收的信息
 */
public class Demo8
{
    public static void main(String[] args) throws InterruptedException
    {
        int i = 0;
        try
        {
            for (int j = 0; j < 100000; j++)
            { // j=100, j=10000
                String.valueOf(j).intern();
                i++;
            }
        }
        catch (Throwable e)
        {
            e.printStackTrace();
        }
        finally
        {
            System.out.println(i);
        }
    }
}

运行后产生的结果:

[GC (Allocation Failure) [PSYoungGen: 2048K->488K(2560K)] 2048K->712K(9728K), 0.0027886 secs] [Times: user=0.00 sys=0.00, real=0.00 secs] 
[GC (Allocation Failure) [PSYoungGen: 2536K->504K(2560K)] 2760K->864K(9728K), 0.0006746 secs] [Times: user=0.00 sys=0.00, real=0.00 secs] 
[GC (Allocation Failure) [PSYoungGen: 2552K->504K(2560K)] 2912K->936K(9728K), 0.0006035 secs] [Times: user=0.00 sys=0.00, real=0.00 secs]  //触发了三次StringTable的垃圾回收
100000
Heap
 PSYoungGen      total 2560K, used 1590K [0x00000000ffd00000, 0x0000000100000000, 0x0000000100000000)
  eden space 2048K, 53% used [0x00000000ffd00000,0x00000000ffe0fa20,0x00000000fff00000)
  from space 512K, 98% used [0x00000000fff00000,0x00000000fff7e010,0x00000000fff80000)
  to   space 512K, 0% used [0x00000000fff80000,0x00000000fff80000,0x0000000100000000)
 ParOldGen       total 7168K, used 432K [0x00000000ff600000, 0x00000000ffd00000, 0x00000000ffd00000)
  object space 7168K, 6% used [0x00000000ff600000,0x00000000ff66c020,0x00000000ffd00000)
 Metaspace       used 3235K, capacity 4496K, committed 4864K, reserved 1056768K
  class space    used 350K, capacity 388K, committed 512K, reserved 1048576K
SymbolTable statistics:
Number of buckets       :     20011 =    160088 bytes, avg   8.000
Number of entries       :     13278 =    318672 bytes, avg  24.000
Number of literals      :     13278 =    567672 bytes, avg  42.753
Total footprint         :           =   1046432 bytes
Average bucket size     :     0.664
Variance of bucket size :     0.663
Std. dev. of bucket size:     0.815
Maximum bucket size     :         6
StringTable statistics:
Number of buckets       :     60013 =    480104 bytes, avg   8.000
Number of entries       :     20948 =    502752 bytes, avg  24.000
Number of literals      :     20948 =   1232496 bytes, avg  58.836 //我们发现产生的字段数量并没有超过100000;说明触发了StringTable的垃圾回收机制。
Total footprint         :           =   2215352 bytes
Average bucket size     :     0.349
Variance of bucket size :     0.364
Std. dev. of bucket size:     0.603
Maximum bucket size     :         4

5.8. StringTable调优
方式一:因为StringTable是由HashTable实现的,所以可以适当增加HashTable桶的个数,来减少字符串放入串池所需要的时间
-XX:StringTableSize=xxxx --调整桶的个数
序号StringTableSize大小运行耗时(单位毫秒)
1100911444
2100091765
3100009430

img

​ 将StringTable桶调大些,示例操作如下:

img

方式二:考虑是否需要将字符串对象入池,可以通过intern方法减少重复入池
/**
 * 演示 intern 减少内存占用
 * -Xsx500m -Xmx500m -XX:+PrintStringTableStatistics -XX:StringTableSize=200000
 */
public class Demo1_24
{
    public static void main(String[] args) throws IOException
    {
        List<String> address = new ArrayList<>();
        System.in.read();
        for (int i = 0; i < 10; i++) //循环10次,会有480万个词;但是有很多是重复的
        {
            try (BufferedReader reader = new BufferedReader(new InputStreamReader(new FileInputStream("linux.words"), "utf-8")))
            {
                String line = null;
                long start = System.nanoTime();
                while (true)
                {
                    line = reader.readLine();
                    if(line == null)
                    {
                        break;
                    }
                    address.add(line.intern()); //是否执行入池操作
                }
                System.out.println("cost:" +(System.nanoTime()-start)/1000000);
            }
        }
        System.in.read();
    }
}

说明一下:linux.words 大约单词量在479829个,上述代码运行结果截图,如下:

读取大约480万单词堆内存占用大小耗时
未放入字符串池约300兆较短
放入字符串池约70兆较长

运行结果1:未放入字符串常量池中,运行情况截图

img

img

运行结果2:放入字符串常量池中,运行情况截图

img

img

6、直接内存

6.1. 定义
  • 属于操作系统,常见于NIO操作时,用于数据缓冲区

  • 分配回收成本较高,但读写性能高

  • 不受JVM内存回收管理

6.2. 文件读写流程
普通内存

需要从用户态向内核态申请资源,即用户态会创建一个java 缓冲区byte[],内核态会创建系统缓冲区。

直接内存

需要从用户态向内核态申请资源,即内核态会创建一块直接内存direct memory,这块direct memory内存可以在用户态、内核态使用。

普通内存读写流程

img

直接内存读写流程;使用了DirectBuffer

img

直接内存是操作系统和Java代码都可以访问的一块区域,无需将代码从系统内存复制到Java堆内存,从而提高了效率

6.3 直接内存与传统方式读取大文件耗时对比案例

接下来,我们将对一个大约1.29G大小的视频文件进行读取并写入指定文件中,即复制。代码如下:

package com.jvm.t05_direct;

import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.IOException;
import java.nio.ByteBuffer;
import java.nio.channels.FileChannel;


public class T01_IoVsDirectBuffer 
{
    static final String FROM = "E:\\Flink CEP.mp4";
    static final String TO = "E:\\a.mp4";
    static final int _1Mb = 1024 * 1024;

    public static void main(String[] args) 
    {
        io(); // io 用时:1535.586957 1766.963399 1359.240226
        directBuffer(); // directBuffer 用时:479.295165 702.291454 562.56592
    }

    private static void directBuffer() 
    {
        long start = System.nanoTime();
        try (FileChannel from = new FileInputStream(FROM).getChannel();
             FileChannel to = new FileOutputStream(TO).getChannel();
            ) 
        {
            ByteBuffer bb = ByteBuffer.allocateDirect(_1Mb);
            while (true) 
            {
                int len = from.read(bb);
                if (len == -1) 
                {
                    break;
                }
                bb.flip();
                to.write(bb);
                bb.clear();
            }
        } 
        catch (IOException e) 
        {
            e.printStackTrace();
        }
        long end = System.nanoTime();
        System.out.println("directBuffer 用时:" + (end - start) / 1000_000.0);
    }

    private static void io() 
    {
        long start = System.nanoTime();
        try (FileInputStream from = new FileInputStream(FROM);
             FileOutputStream to = new FileOutputStream(TO);
            ) 
        {
            byte[] buf = new byte[_1Mb];
            while (true) 
            {
                int len = from.read(buf);
                if (len == -1) 
                {
                    break;
                }
                to.write(buf, 0, len);
            }
        } 
        catch (IOException e) 
        {
            e.printStackTrace();
        }
        long end = System.nanoTime();
        System.out.println("io 用时:" + (end - start) / 1000_000.0);
    }
}

运行耗时对比表如下:

序 号传统方式 IO直接内存directBuffer说明
测试118871.591 ms6335.745 ms没有缓存
测试25710.124 ms5497.707 ms有缓存
测试37355.304 ms5103.806 ms有缓存
6.4 直接内存溢出案例
/**
 * 演示直接内存溢出 java.lang.OutOfMemoryError: Direct buffer memory
 */
 
public class T02_DirectOutOfMemory 
{
    static int _100Mb = 1024 * 1024 * 100;
 
    public static void main(String[] args) 
    {
        List<ByteBuffer> list = new ArrayList<>();
        int i = 0;
        try 
        {
            while (true) 
            {
                ByteBuffer byteBuffer = ByteBuffer.allocateDirect(_100Mb);
                list.add(byteBuffer);
                i++;
            }
        } 
        finally 
        {
            System.out.println(i);
        }
        // 方法区是jvm规范, jdk6 中对方法区的实现称为永久代
        //                  jdk8 对方法区的实现称为元空间
    }
}

直接内存溢出报错:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-2UgINvY1-1669108012080)(C:\Users\vanpersie\AppData\Roaming\Typora\typora-user-images\image-20221113162831887.png)]

6.5. 释放原理

通过代码1我们发现JVM的垃圾回收机制也可以将直接内存回收;

但是直接内存的回收不是通过JVM的垃圾回收来释放的,而是通过unsafe.freeMemory来手动释放

//代码1
public class Demo11
{
    static int _1Gb = 1024 * 1024 * 1024;
    
    public static void main(String[] args) throws IOException
    {
        ByteBuffer byteBuffer = ByteBuffer.allocateDirect(_1Gb);
        System.out.println("分配完毕...");
        System.in.read();
        System.out.println("开始释放...");
        byteBuffer = null;
        System.gc(); 
        System.in.read();
    }
}

通过unsafe.freeMemory释放内存 (我们自己编写的代码)

/**
 * 直接内存分配的底层原理:Unsafe
 */
public class Demo11
{
   static int _1Gb = 1024 * 1024 * 1024;

    public static void main(String[] args) throws IOException 
    {
        Unsafe unsafe = getUnsafe();
        // 分配内存
        long base = unsafe.allocateMemory(_1Gb);
        unsafe.setMemory(base, _1Gb, (byte) 0);
        System.in.read();

        // 释放内存
        unsafe.freeMemory(base);
        System.in.read();
    }

    public static Unsafe getUnsafe() 
    {
        try 
        {
            Field f = Unsafe.class.getDeclaredField("theUnsafe");
            f.setAccessible(true);
            Unsafe unsafe = (Unsafe) f.get(null);
            return unsafe;
        } 
        catch (NoSuchFieldException | IllegalAccessException e) 
        {
            throw new RuntimeException(e);
        }
    }
}

我们再结合源码来分析一下unsafe对直接内存的分配和释放:

//通过ByteBuffer申请1M的直接内存
ByteBuffer byteBuffer = ByteBuffer.allocateDirect(_1M);

申请直接内存,但JVM并不能回收直接内存中的内容,它是如何实现回收的呢?我们通过源码进行分析。

allocateDirect的实现

public static ByteBuffer allocateDirect(int capacity) 
{
    return new DirectByteBuffer(capacity);
}

DirectByteBuffer类

DirectByteBuffer(int cap) {   // package-private
   
    super(-1, 0, cap, cap);
    boolean pa = VM.isDirectMemoryPageAligned();
    int ps = Bits.pageSize();
    long size = Math.max(1L, (long)cap + (pa ? ps : 0));
    Bits.reserveMemory(size, cap);

    long base = 0;
    try {
        base = unsafe.allocateMemory(size); //申请内存
    } catch (OutOfMemoryError x) {
        Bits.unreserveMemory(size, cap);
        throw x;
    }
    unsafe.setMemory(base, size, (byte) 0);
    if (pa && (base % ps != 0)) {
        // Round up to page boundary
        address = base + ps - (base & (ps - 1));
    } else {
        address = base;
    }
    cleaner = Cleaner.create(this, new Deallocator(base, size, cap)); //通过虚引用,来实现直接内存的释放,this为虚引用的实际对象
    att = null;
}

这里调用了一个Cleaner的create方法,且后台线程还会对虚引用的对象监测,如果虚引用的实际对象(这里是DirectByteBuffer)被回收以后,就会调用Cleaner的clean方法,来清除直接内存中占用的内存

public void clean() {
       if (remove(this)) {
           try {
               this.thunk.run(); //调用run方法
           } catch (final Throwable var2) {
               AccessController.doPrivileged(new PrivilegedAction<Void>() {
                   public Void run() {
                       if (System.err != null) {
                           (new Error("Cleaner terminated abnormally", var2)).printStackTrace();
                       }

                       System.exit(1);
                       return null;
                   }
               });
           }
       }
}

对应对象的run方法

public void run() {
    if (address == 0) {
        // Paranoia
        return;
    }
    unsafe.freeMemory(address); //释放直接内存中占用的内存
    address = 0;
    Bits.unreserveMemory(size, capacity);
}
直接内存的回收机制总结
  • 使用了Unsafe类来完成直接内存的分配回收,回收需要主动调用freeMemory方法

  • ByteBuffer的实现内部使用了Cleaner(虚引用)来检测ByteBuffer。一旦ByteBuffer被垃圾回收,那么会由ReferenceHandler来调用Cleaner的clean方法调用freeMemory来释放内存

禁用显示回收

通常情况下我们也可以禁用显示回收; 通过-XX:+DisableExplicitGC 即可实现

/**
 * 禁用显式回收对直接内存的影响
 * 显示回收:让System.gc()失效
 */
public class Demo11
{
    static int _1Gb = 1024 * 1024 * 1024;

    /*
     * -XX:+DisableExplicitGC 显式的
     */
    public static void main(String[] args) throws IOException
    {
        ByteBuffer byteBuffer = ByteBuffer.allocateDirect(_1Gb);
        System.out.println("分配完毕...");
        System.in.read();
        System.out.println("开始释放...");
        byteBuffer = null;
        System.gc(); // 显式的垃圾回收,Full GC
        System.in.read();
    }
}

此时由于我们禁用了显示回收,导致直接内存无法释放。

为了让直接内存可以释放,我们可以采用Unsafe的freeMemory方式来释放直接内存。

static int _1Gb = 1024 * 1024 * 1024;

public static void main(String[] args) throws IOException 
{
    Unsafe unsafe = getUnsafe();
    // 分配内存
    long base = unsafe.allocateMemory(_1Gb);
    unsafe.setMemory(base, _1Gb, (byte) 0);
    System.in.read();

    // 释放内存
    unsafe.freeMemory(base);
    System.in.read();
}

三、垃圾回收

1、如何判断对象可以回收

1.1. 引用计数法

弊端:循环引用时,两个对象的计数都为1,导致两个对象都无法被释放

img

1.2. 可达性分析算法
  • JVM中的垃圾回收器通过可达性分析来探索所有存活的对象

  • 扫描堆中的对象,看能否沿着GC Root对象为起点的引用链找到被扫描对象,如果找不到,则表示可以回收

  • 可以作为GC Root的对象

    • 虚拟机栈(栈帧中的本地变量表)中引用的对象。
    • 方法区中类静态属性引用的对象
    • 方法区中常量引用的对象
    • 本地方法栈中JNI(即一般说的Native方法)引用的对象

    通过案例判断GC Root对象都有哪些:

    //演示GC Root
    public class Demo2_2
    {
        public static void main(String[] args) throws InterruptedException, IOException
        {
            List<Object> list1 = new ArrayList<>();
            list1.add("a");
            list1.add("b");
            System.out.println(1);
            System.in.read();
    
            list1 = null;
            System.out.println(2);
            System.in.read();
            System.out.println("end...");
        }
    }
    
    D:\Ajvm\out\production\jvm\day2>jmap -dump:format=b,live,file=1.bin 6424  --在打印1后截取
    
    D:\Ajvm\out\production\jvm\day2>jmap -dump:format=b,live,file=2.bin 6424  --在打印2后截取
    //可以截取当前时间的内存快照 b是二进制格式;live是存活的对象 file是存放的文件位置
    

    然后我们进入到Eclipice memory analyzer 中查看两个文件

    1.bin中的GC root内容

    [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-icSX3Ddm-1669108012081)(C:\Users\vanpersie\AppData\Roaming\Typora\typora-user-images\image-20221114161554709.png)]

​ 2.bin中的GC root内容

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-9ouRZTqe-1669108012081)(C:\Users\vanpersie\AppData\Roaming\Typora\typora-user-images\image-20221114161608158.png)]

此时发现ArrayList不再存在

1.3. 五种引用

img

强引用(图中的实线表示强引用)

只有GC Root都不引用该对象时,才会回收强引用对象

  • 如上图B、C对象都不引用A1对象时,A1对象才会被回收
软引用

当GC Root指向软引用对象时,在内存不足时(一次垃圾回收后内存仍然不足),会回收软引用所引用的对象

  • 如上图如果B对象不再引用A2对象且内存不足时,软引用所引用的A2对象就会被回收
软引用的使用
/**
 * 演示软引用
 * -Xmx20m -XX:+PrintGCDetails -verbose:gc
 */
public class Demo2_3 
{
    private static final int _4MB = 4 * 1024 * 1024;
    
    public static void main(String[] args) throws IOException
    {
        /*此时list对byte采用的是强引用;由于我们已经设置了最大大小,如果使用强引用执行下面代码就会导致程序报错。
        此时我们可以使用软引用进行回收,以此来释放内存,当再次使用时再加载*/
        /*List<byte[]> list = new ArrayList<>();
        for (int i = 0; i < 5; i++) {
            list.add(new byte[_4MB]);
        }
        System.in.read();*/
        soft();
    }

    public static void soft()
    {
        // list --> SoftReference --> byte[]
        /*此时 list对SoftReference是强引用;SoftReference对byte是软引用*/
        List<SoftReference<byte[]>> list = new ArrayList<>();
        for (int i = 0; i < 5; i++)
        {
            SoftReference<byte[]> ref = new SoftReference<>(new byte[_4MB]);
            System.out.println(ref.get());
            list.add(ref);
            System.out.println(list.size());
        }

        System.out.println("循环结束:" + list.size());
        for (SoftReference<byte[]> ref : list)
        {
            System.out.println(ref.get());
        }
    }
}

运行所展示的结果,发现只有最后一次循环在内存中

[B@7f31245a
1
[B@6d6f6e28
2
[B@135fbaa4
3
[GC (Allocation Failure) [PSYoungGen: 1946K->504K(6144K)] 14234K->13048K(19968K), 0.0026561 secs] [Times: user=0.00 sys=0.00, real=0.00 secs] //进行了一次垃圾回收
[B@45ee12a7
4
[GC (Allocation Failure) --[PSYoungGen: 4712K->4712K(6144K)] 17256K->17296K(19968K), 0.0006782 secs] [Times: user=0.00 sys=0.00, real=0.00 secs]  //进行了一次垃圾回收,发现不够
 
[Full GC (Ergonomics) [PSYoungGen: 4712K->4515K(6144K)] [ParOldGen: 12584K->12518K(13824K)] 17296K->17033K(19968K), [Metaspace: 3230K->3230K(1056768K)], 0.0034193 secs] [Times: user=0.00 sys=0.00, real=0.00 secs] //进行了一次Full GC垃圾回收,发现仍然不够
[GC (Allocation Failure) --[PSYoungGen: 4515K->4515K(6144K)] 17033K->17101K(19968K), 0.0003413 secs] [Times: user=0.00 sys=0.00, real=0.00 secs] 
[Full GC (Allocation Failure) [PSYoungGen: 4515K->0K(6144K)] [ParOldGen: 12586K->631K(8704K)] 17101K->631K(14848K), [Metaspace: 3230K->3230K(1056768K)], 0.0038976 secs] [Times: user=0.00 sys=0.00, real=0.00 secs] 
 //由于内存不足,回收之前四次的软引用对象以此来释放内存
 
[B@330bedb4
5
 
循环结束:5
null
null
null
null
[B@330bedb4
Heap
 PSYoungGen      total 6144K, used 4264K [0x00000000ff980000, 0x0000000100000000, 0x0000000100000000)
  eden space 5632K, 75% used [0x00000000ff980000,0x00000000ffdaa2d8,0x00000000fff00000)
  from space 512K, 0% used [0x00000000fff00000,0x00000000fff00000,0x00000000fff80000)
  to   space 512K, 0% used [0x00000000fff80000,0x00000000fff80000,0x0000000100000000)
 ParOldGen       total 8704K, used 631K [0x00000000fec00000, 0x00000000ff480000, 0x00000000ff980000)
  object space 8704K, 7% used [0x00000000fec00000,0x00000000fec9dcd0,0x00000000ff480000)
 Metaspace       used 3238K, capacity 4500K, committed 4864K, reserved 1056768K
  class space    used 351K, capacity 388K, committed 512K, reserved 1048576K

软引用配合引用队列使用

如果在垃圾回收时发现内存不足,在回收软引用所指向的对象时,软引用本身不会被清理

如果想要清理软引用,需要使用引用队列

/**
 * 演示软引用, 配合引用队列
 */
public class Demo2_4
{
    private static final int _4MB = 4 * 1024 * 1024;

    public static void main(String[] args)
    {
        List<SoftReference<byte[]>> list = new ArrayList<>();

        // 引用队列
        ReferenceQueue<byte[]> queue = new ReferenceQueue<>();

        for (int i = 0; i < 5; i++)
        {
            // 关联了引用队列,当软引用所关联的 byte[]被回收时,软引用自己会加入到 queue 中去
            SoftReference<byte[]> ref = new SoftReference<>(new byte[_4MB], queue);
            System.out.println(ref.get());
            list.add(ref);
            System.out.println(list.size());
        }

        // 从队列中获取无用的 软引用对象,并移除
        Reference<? extends byte[]> poll = queue.poll();
        while( poll != null)
        {
            list.remove(poll);
            poll = queue.poll();
        }

        System.out.println("===========================");
        for (SoftReference<byte[]> reference : list)
        {
            System.out.println(reference.get());
        }
    }
}

运行结果发现最终集合中仅有第五次循环所添加的内容

[B@61064425
1
[B@7b1d7fff
2
[B@299a06ac
3
[GC (Allocation Failure) [PSYoungGen: 2273K->512K(6144K)] 14561K->13068K(19968K), 0.0008631 secs] [Times: user=0.00 sys=0.00, real=0.00 secs] 
[B@383534aa
4
[GC (Allocation Failure) --[PSYoungGen: 4720K->4720K(6144K)] 17277K->17329K(19968K), 0.0005521 secs] [Times: user=0.00 sys=0.00, real=0.00 secs] 
[Full GC (Ergonomics) [PSYoungGen: 4720K->4522K(6144K)] [ParOldGen: 12608K->12552K(13824K)] 17329K->17075K(19968K), [Metaspace: 3058K->3058K(1056768K)], 0.0050865 secs] [Times: user=0.00 sys=0.00, real=0.01 secs] 
[GC (Allocation Failure) --[PSYoungGen: 4522K->4522K(6144K)] 17075K->17107K(19968K), 0.0004535 secs] [Times: user=0.00 sys=0.00, real=0.00 secs] 
[Full GC (Allocation Failure) [PSYoungGen: 4522K->0K(6144K)] [ParOldGen: 12584K->675K(8704K)] 17107K->675K(14848K), [Metaspace: 3058K->3058K(1056768K)], 0.0041101 secs] [Times: user=0.00 sys=0.00, real=0.00 secs] 
[B@6bc168e5
5
===========================
[B@6bc168e5  //仅有第五次循环添加的内容
Heap
 PSYoungGen      total 6144K, used 4377K [0x00000000ff980000, 0x0000000100000000, 0x0000000100000000)
  eden space 5632K, 77% used [0x00000000ff980000,0x00000000ffdc6418,0x00000000fff00000)
  from space 512K, 0% used [0x00000000fff00000,0x00000000fff00000,0x00000000fff80000)
  to   space 512K, 0% used [0x00000000fff80000,0x00000000fff80000,0x0000000100000000)
 ParOldGen       total 8704K, used 675K [0x00000000fec00000, 0x00000000ff480000, 0x00000000ff980000)
  object space 8704K, 7% used [0x00000000fec00000,0x00000000feca8ed0,0x00000000ff480000)
 Metaspace       used 3068K, capacity 4556K, committed 4864K, reserved 1056768K
  class space    used 325K, capacity 392K, committed 512K, reserved 1048576K

**大概思路为:**查看引用队列中有无软引用,如果有,则将该软引用从存放它的集合中移除(这里为一个list集合)

弱引用

只有弱引用引用该对象时,在垃圾回收时,无论内存是否充足,都会回收弱引用所引用的对象

  • 如上图如果B对象不再引用A3对象,则A3对象会被回收

弱引用的使用和软引用类似,只是将 SoftReference 换为了 WeakReference

弱引用的使用
/**
 * 演示弱引用
 * -Xmx20m -XX:+PrintGCDetails -verbose:gc
 */
public class Demo2_5
{
    private static final int _4MB = 4 * 1024 * 1024;

    public static void main(String[] args)
    {
        //  list --> WeakReference --> byte[]
        List<WeakReference<byte[]>> list = new ArrayList<>();
        for (int i = 0; i < 10; i++)
        {
            WeakReference<byte[]> ref = new WeakReference<>(new byte[_4MB]);
            list.add(ref);
            for (WeakReference<byte[]> w : list)
            {
                System.out.print(w.get()+" ");
            }
            System.out.println();
        }

        System.out.println("循环结束:" + list.size());
        System.out.println("=======================================");
    }
}

运行结果

[B@7f31245a 
[B@7f31245a [B@6d6f6e28 
[B@7f31245a [B@6d6f6e28 [B@135fbaa4 
[GC (Allocation Failure) [PSYoungGen: 2058K->488K(6144K)] 14346K->13004K(19968K), 0.0007146 secs] [Times: user=0.00 sys=0.00, real=0.00 secs] 
[B@7f31245a [B@6d6f6e28 [B@135fbaa4 [B@45ee12a7 
[GC (Allocation Failure) [PSYoungGen: 4696K->504K(6144K)] 17212K->13116K(19968K), 0.0005101 secs] [Times: user=0.02 sys=0.00, real=0.00 secs] 
[B@7f31245a [B@6d6f6e28 [B@135fbaa4 null [B@330bedb4 
[GC (Allocation Failure) [PSYoungGen: 4712K->504K(6144K)] 17324K->13164K(19968K), 0.0003153 secs] [Times: user=0.00 sys=0.00, real=0.00 secs] 
[B@7f31245a [B@6d6f6e28 [B@135fbaa4 null null [B@2503dbd3 
[GC (Allocation Failure) [PSYoungGen: 4711K->504K(6144K)] 17371K->13252K(19968K), 0.0002964 secs] [Times: user=0.00 sys=0.00, real=0.00 secs] 
[B@7f31245a [B@6d6f6e28 [B@135fbaa4 null null null [B@4b67cf4d 
[GC (Allocation Failure) [PSYoungGen: 4710K->488K(6144K)] 17458K->13260K(19968K), 0.0003348 secs] [Times: user=0.00 sys=0.00, real=0.00 secs] 
[B@7f31245a [B@6d6f6e28 [B@135fbaa4 null null null null [B@7ea987ac 
[GC (Allocation Failure) [PSYoungGen: 4694K->504K(5120K)] 17466K->13316K(18944K), 0.0002603 secs] [Times: user=0.00 sys=0.00, real=0.00 secs] 
[B@7f31245a [B@6d6f6e28 [B@135fbaa4 null null null null null [B@12a3a380 
[GC (Allocation Failure) [PSYoungGen: 4690K->64K(5632K)] 17502K->13332K(19456K), 0.0003107 secs] [Times: user=0.00 sys=0.00, real=0.00 secs] 
[Full GC (Ergonomics) [PSYoungGen: 64K->0K(5632K)] [ParOldGen: 13268K->649K(8704K)] 13332K->649K(14336K), [Metaspace: 3230K->3230K(1056768K)], 0.0038648 secs] [Times: user=0.00 sys=0.00, real=0.00 secs] 
null null null null null null null null null [B@29453f44 //之前的弱引用全部被回收掉
循环结束:10
=======================================
Heap
 PSYoungGen      total 5632K, used 4278K [0x00000000ff980000, 0x0000000100000000, 0x0000000100000000)
  eden space 4608K, 92% used [0x00000000ff980000,0x00000000ffdadb20,0x00000000ffe00000)
  from space 1024K, 0% used [0x00000000ffe00000,0x00000000ffe00000,0x00000000fff00000)
  to   space 1024K, 0% used [0x00000000fff00000,0x00000000fff00000,0x0000000100000000)
 ParOldGen       total 8704K, used 649K [0x00000000fec00000, 0x00000000ff480000, 0x00000000ff980000)
  object space 8704K, 7% used [0x00000000fec00000,0x00000000feca2610,0x00000000ff480000)
 Metaspace       used 3237K, capacity 4500K, committed 4864K, reserved 1056768K
  class space    used 350K, capacity 388K, committed 512K, reserved 1048576K
虚引用

当虚引用对象所引用的对象被回收以后,虚引用对象就会被放入引用队列中,调用虚引用的方法

  • 虚引用的一个体现是释放直接内存所分配的内存,当引用的对象ByteBuffer被垃圾回收以后,虚引用对象Cleaner就会被放入引用队列中,然后调用Cleaner的clean方法来释放直接内存
  • 如上图,B对象不再引用ByteBuffer对象,ByteBuffer就会被回收。但是直接内存中的内存还未被回收。这时需要将虚引用对象Cleaner放入引用队列中,然后调用它的clean方法来释放直接内存
终结器引用

所有的类都继承自Object类,Object类有一个finalize方法。当某个对象不再被其他的对象所引用时,会先将终结器引用对象放入引用队列中,然后根据终结器引用对象找到它所引用的对象,然后调用该对象的finalize方法。调用以后,该对象就可以被垃圾回收了

  • 如上图,B对象不再引用A4对象。这是终结器对象就会被放入引用队列中,引用队列会根据它,找到它所引用的对象。然后调用被引用对象的finalize方法。调用以后,该对象就可以被垃圾回收了
引用队列
  • 软引用和弱引用可以配合引用队列
    • 弱引用虚引用所引用的对象被回收以后,会将这些引用放入引用队列中,方便一起回收这些软/弱引用对象
  • 虚引用和终结器引用必须配合引用队列
    • 虚引用和终结器引用在使用时会关联一个引用队列

2、垃圾回收算法

2.1. 标记-清除

img

定义:标记清除算法顾名思义,是指在虚拟机执行垃圾回收的过程中,先采用标记算法确定可回收对象,然后垃圾收集器根据标识清除相应的内容,给堆内存腾出相应的空间

  • 这里的腾出内存空间并不是将内存空间的字节清0,而是记录下这段内存的起始结束地址,下次分配内存的时候,会直接覆盖这段内存

缺点容易产生大量的内存碎片,可能无法满足大对象的内存分配,一旦导致无法分配对象,那就会导致jvm启动gc,一旦启动gc,我们的应用程序就会暂停,这就导致应用的响应速度变慢

2.2. 标记-整理

img

标记-整理 会将不被GC Root引用的对象回收,清楚其占用的内存空间。然后整理剩余的对象,可以有效避免因内存碎片而导致的问题,但是因为整体需要消耗一定的时间,所以效率较低

缺点效率较低

2.3. 复制

img

img

img

img

将内存分为等大小的两个区域,FROM和TO(TO中为空)。先将被GC Root引用的对象从FROM放入TO中,再回收不被GC Root引用的对象。然后交换FROM和TO。这样也可以避免内存碎片的问题,但是会占用双倍的内存空间。

缺点:会占用双倍内存空间

3、分代回收

新生代:主要是用来存放新生的,内存较小的对象;垃圾回收频率高(如每家每户的日常垃圾,体量小,需要及时回收)

老年代:存放生命周期比较长,大小较大的对象;垃圾回收频率低(如家里破旧的椅子,可以等到家里空间不足时再扔掉)

img

3.1. 回收流程

新创建的对象都被放在了新生代的伊甸园

img

当伊甸园中的内存不足时,就会进行一次垃圾回收,这时的回收叫做 Minor GC (新生代的垃圾回收名称)

img

Minor GC 会将伊甸园和幸存区FROM存活的对象复制到 幸存区 TO中, 并让其寿命加1

img

然后再交换两个幸存区

img

再次创建对象,若新生代的伊甸园又满了,则会再次触发 Minor GC(会触发 stop the world, 暂停其他用户线程,只让垃圾回收线程工作;但是时间较短),这时不仅会回收伊甸园中的垃圾,还会回收幸存区中的垃圾,再将不需要回收的对象复制到幸存区TO中。回收以后会交换两个幸存区,并让幸存区中的对象寿命加1

img

如果幸存区中的对象的寿命超过某个阈值(最大为15,4bit),就会被放入老年代

img

如果新生代老年代中的内存都满了,就会先触发Minor GC。如果之后空间仍然不足,再触发Full GC (会触发stop the world, 暂停其他用户线程,只让垃圾回收线程工作;但是时间较长),扫描新生代和老年代中所有不再使用的对象并回收

3.2. GC 分析
3.2.1. 相关VM参数
含义参数
堆初始大小-Xms
堆最大大小-Xmx 或 -XX:MaxHeapSize=size
新生代大小-Xmn 或 (-XX:NewSize=size + -XX:MaxNewSize=size )
幸存区比例(动态)-XX:InitialSurvivorRatio=ratio 和 -XX:+UseAdaptiveSizePolicy
幸存区比例-XX:SurvivorRatio=ratio
晋升阈值-XX:MaxTenuringThreshold=threshold
晋升详情-XX:+PrintTenuringDistribution
GC详情-XX:+PrintGCDetails -verbose:gc
FullGC 前 MinorGC-XX:+ScavengeBeforeFullGC
3.2.2. 通过代码分析回收流程
/**
 *  演示内存的分配策略
 */
public class Main 
{
    private static final int _512KB = 512 * 1024;
    private static final int _1MB = 1024 * 1024;
    private static final int _6MB = 6 * 1024 * 1024;
    private static final int _7MB = 7 * 1024 * 1024;
    private static final int _8MB = 8 * 1024 * 1024;
    // -Xms20M -Xmx20M -Xmn10M -XX:+UseSerialGC -XX:+PrintGCDetails -verbose:gc -XX:-ScavengeBeforeFullGC
    public static void main(String[] args) throws InterruptedException 
    {
        ArrayList<byte[]> list=new ArrayList<>();
        list.add(new byte[_7MB]);
        //list.add(new byte[_512KB]);
        //list.add(new byte[_512KB]);
    }
}

添加7MB时运行结果

[GC (Allocation Failure) [DefNew: 2168K->618K(9216K), 0.0012407 secs] 2168K->618K(19456K), 0.0012704 secs] [Times: user=0.00 sys=0.00, real=0.00 secs]  //进行了一次垃圾回收
Heap //我们最开始给新生代分配了10M的空间,但是此处显示的时9.2M。是因为会扣除一部分幸存区to的空间,幸存区to空间默认是不放东西的
 def new generation   total 9216K, used 8169K [0x00000000fec00000, 0x00000000ff600000, 0x00000000ff600000) //新生代
  eden space 8192K,  92% used [0x00000000fec00000, 0x00000000ff35faf8, 0x00000000ff400000) //伊甸园
  from space 1024K,  60% used [0x00000000ff500000, 0x00000000ff59abe8, 0x00000000ff600000) //幸存区from
  to   space 1024K,   0% used [0x00000000ff400000, 0x00000000ff400000, 0x00000000ff500000) //幸存区to
 tenured generation   total 10240K, used 0K [0x00000000ff600000, 0x0000000100000000, 0x0000000100000000) //老年代
   the space 10240K,   0% used [0x00000000ff600000, 0x00000000ff600000, 0x00000000ff600200, 0x0000000100000000)
 Metaspace       used 3226K, capacity 4496K, committed 4864K, reserved 1056768K //元空间
  class space    used 349K, capacity 388K, committed 512K, reserved 1048576K

再添加512KB时的运行结果

[GC (Allocation Failure) [DefNew: 2168K->620K(9216K), 0.0015122 secs] 2168K->620K(19456K), 0.0015491 secs] [Times: user=0.00 sys=0.00, real=0.00 secs] 
Heap
 def new generation   total 9216K, used 8683K [0x00000000fec00000, 0x00000000ff600000, 0x00000000ff600000)
  eden space 8192K,  98% used [0x00000000fec00000, 0x00000000ff3dfb18, 0x00000000ff400000) //此时发现伊甸园中的空间即将满
  from space 1024K,  60% used [0x00000000ff500000, 0x00000000ff59b188, 0x00000000ff600000)
  to   space 1024K,   0% used [0x00000000ff400000, 0x00000000ff400000, 0x00000000ff500000)
 tenured generation   total 10240K, used 0K [0x00000000ff600000, 0x0000000100000000, 0x0000000100000000)
   the space 10240K,   0% used [0x00000000ff600000, 0x00000000ff600000, 0x00000000ff600200, 0x0000000100000000)
 Metaspace       used 3232K, capacity 4496K, committed 4864K, reserved 1056768K
  class space    used 350K, capacity 388K, committed 512K, reserved 1048576K

再添加512KB时的运行结果

[GC (Allocation Failure) [DefNew: 2005K->617K(9216K), 0.0011718 secs] 2005K->617K(19456K), 0.0011981 secs] [Times: user=0.00 sys=0.00, real=0.00 secs] 
[GC (Allocation Failure) [DefNew: 8625K->519K(9216K), 0.0051663 secs] 8625K->8294K(19456K), 0.0051942 secs] [Times: user=0.00 sys=0.02, real=0.01 secs] //进行了两次垃圾回收
Heap
 def new generation   total 9216K, used 1196K [0x00000000fec00000, 0x00000000ff600000, 0x00000000ff600000)
  eden space 8192K,   8% used [0x00000000fec00000, 0x00000000feca9140, 0x00000000ff400000)
  from space 1024K,  50% used [0x00000000ff400000, 0x00000000ff481f68, 0x00000000ff500000)
  to   space 1024K,   0% used [0x00000000ff500000, 0x00000000ff500000, 0x00000000ff600000)
 tenured generation   total 10240K, used 7774K [0x00000000ff600000, 0x0000000100000000, 0x0000000100000000)
   the space 10240K,  75% used [0x00000000ff600000, 0x00000000ffd97bc0, 0x00000000ffd97c00, 0x0000000100000000) //因为新生代中空间不足,此时一部分新生代直接进入到了老年代中
 Metaspace       used 3226K, capacity 4496K, committed 4864K, reserved 1056768K
  class space    used 349K, capacity 388K, committed 512K, reserved 1048576K
3.2.3. 大对象处理策略

当遇到一个较大的对象时,就算新生代的伊甸园为空,也无法容纳该对象时,会将该对象直接晋升为老年代

/**
 *  演示内存的分配策略
 */
public class Demo2_1
{
    private static final int _512KB = 512 * 1024;
    private static final int _1MB = 1024 * 1024;
    private static final int _6MB = 6 * 1024 * 1024;
    private static final int _7MB = 7 * 1024 * 1024;
    private static final int _8MB = 8 * 1024 * 1024;

    // -Xms20M -Xmx20M -Xmn10M -XX:+UseSerialGC -XX:+PrintGCDetails -verbose:gc -XX:-ScavengeBeforeFullGC
    public static void main(String[] args) throws InterruptedException
    {
         ArrayList<byte[]> list = new ArrayList<>();
         list.add(new byte[_8MB]);
    }
}

运行结果如下,并没有触发垃圾回收,而是直接晋升为老年代

Heap
 def new generation   total 9216K, used 2169K [0x00000000fec00000, 0x00000000ff600000, 0x00000000ff600000)
  eden space 8192K,  26% used [0x00000000fec00000, 0x00000000fee1e538, 0x00000000ff400000)
  from space 1024K,   0% used [0x00000000ff400000, 0x00000000ff400000, 0x00000000ff500000)
  to   space 1024K,   0% used [0x00000000ff500000, 0x00000000ff500000, 0x00000000ff600000)
 tenured generation   total 10240K, used 8192K [0x00000000ff600000, 0x0000000100000000, 0x0000000100000000)
   the space 10240K,  80% used [0x00000000ff600000, 0x00000000ffe00010, 0x00000000ffe00200, 0x0000000100000000)
 Metaspace       used 3231K, capacity 4496K, committed 4864K, reserved 1056768K
  class space    used 350K, capacity 388K, committed 512K, reserved 1048576K

再添加8MB时,会导致堆的空间不足。先触发一次Minor GC 和 一次Full GC 垃圾回收,但回收后的空间仍然不足,会抛出堆空间不足的异常

[GC (Allocation Failure) [DefNew: 2168K->649K(9216K), 0.0016210 secs][Tenured: 8192K->8840K(10240K), 0.0016607 secs] 10360K->8840K(19456K), [Metaspace: 3225K->3225K(1056768K)], 0.0033266 secs] [Times: user=0.00 sys=0.00, real=0.00 secs] 
[Full GC (Allocation Failure) [Tenured: 8840K->8822K(10240K), 0.0009837 secs] 8840K->8822K(19456K), [Metaspace: 3225K->3225K(1056768K)], 0.0009982 secs] [Times: user=0.00 sys=0.00, real=0.00 secs] 
Heap
 def new generation   total 9216K, used 382K [0x00000000fec00000, 0x00000000ff600000, 0x00000000ff600000)
  eden space 8192K,   4% used [0x00000000fec00000, 0x00000000fec5fac8, 0x00000000ff400000)
  from space 1024K,   0% used [0x00000000ff500000, 0x00000000ff500000, 0x00000000ff600000)
  to   space 1024K,   0% used [0x00000000ff400000, 0x00000000ff400000, 0x00000000ff500000)
 tenured generation   total 10240K, used 8822K [0x00000000ff600000, 0x0000000100000000, 0x0000000100000000)
   the space 10240K,  86% used [0x00000000ff600000, 0x00000000ffe9d9b0, 0x00000000ffe9da00, 0x0000000100000000)
 Metaspace       used 3257K, capacity 4496K, committed 4864K, reserved 1056768K
  class space    used 353K, capacity 388K, committed 512K, reserved 1048576K
                                
Exception in thread "main" java.lang.OutOfMemoryError: Java heap space //程序报错
	at day2.Demo2_1.main(Demo2_1.java:21)
3.2.4. 线程内存溢出

可能存在的误区;假设堆空间不足出现在一个子线程中,主线程是否会受到影响呢?答案是不会。

某个线程的内存溢出了而抛异常(out of memory),不会让其他的线程结束运行

这是因为当一个线程抛出OOM异常后它所占据的内存资源会全部被释放掉,从而不会影响其他线程的运行,进程依然正常

演示代码如下:

public class Demo2_1
{
    private static final int _512KB = 512 * 1024;
    private static final int _1MB = 1024 * 1024;
    private static final int _6MB = 6 * 1024 * 1024;
    private static final int _7MB = 7 * 1024 * 1024;
    private static final int _8MB = 8 * 1024 * 1024;

    // -Xms20M -Xmx20M -Xmn10M -XX:+UseSerialGC -XX:+PrintGCDetails -verbose:gc -XX:-ScavengeBeforeFullGC
    public static void main(String[] args) throws InterruptedException
    {
        new Thread(() ->
        {
            ArrayList<byte[]> list = new ArrayList<>();
            list.add(new byte[_8MB]);
            list.add(new byte[_8MB]);
        }).start();

        System.out.println("sleep....");
        Thread.sleep(1000L);
    }
}

运行结果

sleep....
[GC (Allocation Failure) [DefNew: 4515K->854K(9216K), 0.0019388 secs][Tenured: 8192K->9044K(10240K), 0.0021212 secs] 12707K->9044K(19456K), [Metaspace: 4155K->4155K(1056768K)], 0.0041084 secs] [Times: user=0.01 sys=0.00, real=0.00 secs] 
[Full GC (Allocation Failure) [Tenured: 9044K->8988K(10240K), 0.0014433 secs] 9044K->8988K(19456K), [Metaspace: 4155K->4155K(1056768K)], 0.0014670 secs] [Times: user=0.00 sys=0.00, real=0.00 secs] 
    
Exception in thread "Thread-0" java.lang.OutOfMemoryError: Java heap space //子线程发生了堆空间不足的报错
	at day2.Demo2_1.lambda$main$0(Demo2_1.java:23)
	at day2.Demo2_1$$Lambda$1/1023892928.run(Unknown Source)
	at java.lang.Thread.run(Thread.java:748)
    
Heap	//但是主线程并没有受到影响
 def new generation   total 9216K, used 1377K [0x00000000fec00000, 0x00000000ff600000, 0x00000000ff600000)
  eden space 8192K,  16% used [0x00000000fec00000, 0x00000000fed586c0, 0x00000000ff400000)
  from space 1024K,   0% used [0x00000000ff500000, 0x00000000ff500000, 0x00000000ff600000)
  to   space 1024K,   0% used [0x00000000ff400000, 0x00000000ff400000, 0x00000000ff500000)
 tenured generation   total 10240K, used 8988K [0x00000000ff600000, 0x0000000100000000, 0x0000000100000000)
   the space 10240K,  87% used [0x00000000ff600000, 0x00000000ffec7218, 0x00000000ffec7400, 0x0000000100000000)
 Metaspace       used 4678K, capacity 4744K, committed 4992K, reserved 1056768K
  class space    used 519K, capacity 560K, committed 640K, reserved 1048576K

4、垃圾回收器

4.1. 相关概念

并行收集:指多条垃圾收集线程并行工作,但此时用户线程仍处于等待状态

并发收集:指用户线程与垃圾收集线程同时工作(不一定是并行的可能会交替执行)。用户程序在继续运行,而垃圾收集程序运行在另一个CPU上

吞吐量:即CPU用于运行用户代码的时间与CPU总消耗时间的比值(吞吐量 = 运行用户代码时间 / ( 运行用户代码时间 + 垃圾收集时间 )),也就是。例如:虚拟机共运行100分钟,垃圾收集器花掉1分钟,那么吞吐量就是99%

垃圾回收器分为串行垃圾回收器、吞吐量优先垃圾回收器和响应时间优先垃圾回收器;

4.2. 串行垃圾回收器
  • 单线程

  • 内存较小,个人电脑(CPU核数较少)

    XX:+UseSerialGC = Serial + SerialOld,新生代**-Serial** ,老年代-SerialOld

img

安全点:让其他线程都在这个点停下来,以免垃圾回收时移动对象地址,使得其他线程找不到被移动的对象

因为是串行的,所以只有一个垃圾回收线程。且在该线程执行回收工作时,其他线程进入阻塞状态

Serial 收集器

Serial收集器是最基本的、发展历史最悠久的收集器

特点:单线程、简单高效(与其他收集器的单线程相比),采用复制算法。对于限定单个CPU的环境来说,Serial收集器由于没有线程交互的开销,专心做垃圾收集自然可以获得最高的单线程收集效率。收集器进行垃圾回收时,必须暂停其他所有的工作线程,直到它结束(Stop The World)

ParNew 收集器

ParNew收集器其实就是Serial收集器的多线程版本

特点:多线程、ParNew收集器默认开启的收集线程数与CPU的数量相同,在CPU非常多的环境中,可以使用-XX:ParallelGCThreads参数来限制垃圾收集的线程数。和Serial收集器一样存在Stop The World问题

Serial Old 收集器

Serial Old是Serial收集器的老年代版本

特点:同样是单线程收集器,采用标记-整理算法

4.3. 吞吐量优先垃圾回收器
  • 多线程
  • 堆内存较大,多核CPU
  • 单位时间内,STW(stop the world,停掉其他所有工作线程)总时间最短
  • JDK1.8默认使用的垃圾回收器

img

Parallel Scavenge 收集器

与吞吐量关系密切,故也称为吞吐量优先收集器

特点:属于新生代收集器也是采用复制算法的收集器(用到了新生代的幸存区),又是并行的多线程收集器(与ParNew收集器类似)

该收集器的目标是达到一个可控制的吞吐量。还有一个值得关注的点是:GC自适应调节策略(与ParNew收集器最重要的一个区别)

GC自适应调节策略:Parallel Scavenge收集器可设置-XX:+UseAdptiveSizePolicy参数。当开关打开时不需要手动指定新生代的大小(-Xmn)、Eden与Survivor区的比例(-XX:SurvivorRation)、晋升老年代的对象年龄(-XX:PretenureSizeThreshold)等,虚拟机会根据系统的运行状况收集性能监控信息,动态设置这些参数以提供最优的停顿时间和最高的吞吐量,这种调节方式称为GC的自适应调节策略

Parallel Scavenge收集器使用两个参数控制吞吐量:

  • XX:MaxGCPauseMillis 控制最大的垃圾收集停顿时间
  • XX:GCRatio 直接设置吞吐量的大小
Parallel Old 收集器

是Parallel Scavenge收集器的老年代版本

特点:多线程,采用标记-整理算法(老年代没有幸存区)

4.4. 响应时间优先垃圾回收器
  • 多线程
  • 堆内存较大,多核CPU
  • 尽可能让单次STW时间最短(尽量不影响其他线程运行)

img

CMS 收集器

Concurrent Mark Sweep,一种以获取最短回收停顿时间为目标的老年代收集器

特点:基于标记-清除算法实现。并发收集、低停顿,但是会产生内存碎片

应用场景:适用于注重服务的响应速度,希望系统停顿时间最短,给用户带来更好的体验等场景下。如web程序、b/s服务

CMS收集器的运行过程分为下列4步:

初始标记:标记GC Roots能直接到的对象。速度很快但是仍存在Stop The World问题

并发标记:进行GC Roots Tracing 的过程,找出存活对象且用户线程可并发执行

重新标记:为了修正并发标记期间因用户程序继续运行而导致标记产生变动的那一部分对象的标记记录。仍然存在Stop The World问题

并发清除:对标记的对象进行清除回收

CMS收集器的内存回收过程是与用户线程一起并发执行

4.5. G1垃圾回收器
(1). 定义

Garbage First

JDK 9以后默认使用,而且替代了CMS 收集器

img

(2). 适用场景
  • 同时注重吞吐量和低延迟(响应时间)
  • 超大堆内存(内存大的),会将堆内存划分为多个大小相等的区域(Region)
  • 整体上是标记-整理算法,两个区域之间是复制算法

相关参数:JDK8 并不是默认开启的,所需要参数开启

-XX:+UseG1GC //即可开启G1垃圾回收器
-XX:G1HeapRegionSize=size //把堆分成多少个大小相等的区域; 但是大小必须是1、2、4、8、16等    
-XX:MaxGCPauseMillis=time //每次垃圾回收最大暂停毫秒数

img

(3). G1垃圾回收阶段

img

新生代伊甸园垃圾回收—–>内存不足,新生代回收+并发标记—–>回收新生代伊甸园、幸存区、老年代内存——>新生代伊甸园垃圾回收(重新开始)

(4). Young Collection

分区算法region

分代是按对象的生命周期划分,

分区则是将堆空间划分连续几个不同小区间,每一个小区间独立回收,可以控制一次回收多少个小区间,方便控制 GC 产生的停顿时间

每个区域都可以作为 伊甸园、幸存区和老年代

E:伊甸园 S:幸存区 O:老年代

  • 存在STW

img

当内存不足时,开始进行新生代回收;采取的是复制算法,将剩余的对象复制到幸存区

img

当幸存区对象过多或幸存区中的某个对象超过最大的阈值。又会触发垃圾回收,此时幸存区中的一部分内容会复制到老年代中。

img

(5). Young Collection + CM

CM:并发标记

  • 在 Young GC 时会对 GC Root 进行初始标记

  • 在老年代占用堆内存的比例达到阈值时,会发生并发标记(不会STW),阈值可以根据用户来进行设定, 参数如下

    -XX:InitiatingHeapOccupancyPercent=percent //(默认45%)老年代占用堆内存的比例达到45%时,会发生并发标记
    

img

(6). Mixed Collection

会对E S O 进行全面的回收; 会经历最终标记和拷贝存活两个阶段

  • 最终标记(remark): 对并发标记时其它线程正在运行时漏掉的对象进行标记;会STW
  • 拷贝存活(Evacuation): 将一些有价值的老年代拷贝到新的老年代区域中,让它们存活;会STW
-XX:MaxGCPauseMills:xxx //用于指定最长的停顿时间

:为什么有的老年代被拷贝了,有的没拷贝?

因为指定了最大停顿时间,如果对所有老年代都进行回收,耗时可能过高。为了保证时间不超过设定的停顿时间,会回收最有价值的老年代(回收后,能够得到更多内存)

img

(7). 四种垃圾回收器中 Full GC 的不同
SerialGC
	新生代内存不足发生的垃圾收集 - minor gc
	老年代内存不足发生的垃圾收集 - full gc
	
ParallelGC
	新生代内存不足发生的垃圾收集 - minor gc
	老年代内存不足发生的垃圾收集 - full gc
	
CMS
	新生代内存不足发生的垃圾收集 - minor gc
	老年代内存不足(老年代所占内存超过阈值)
			如果垃圾产生速度慢于垃圾回收速度,不会触发Full GC,还是并发地进行清理
			如果垃圾产生速度快于垃圾回收速度,便会触发Full GC
G1
	新生代内存不足发生的垃圾收集 - minor gc
	老年代内存不足(老年代所占内存超过阈值)
			如果垃圾产生速度慢于垃圾回收速度,不会触发Full GC,还是并发地进行清理
			如果垃圾产生速度快于垃圾回收速度,便会触发Full GC
(8). Young Collection 跨代引用

新生代在寻找GC Root进行筛选存活对象的时候,会涉及到去老年代中寻找根对象的问题。

但老年代中存活的对象非常多,如果我们遍历整个老年代寻找根对象效率非常低。所以这里我们采取的是卡表的技术将老年代再进一步细分。

  • 新生代回收的跨代引用(老年代引用新生代)问题

img

  • 卡表与Remembered Set
    • 卡表:将老年代分为多个区域(一个区域512K)
    • Remembered Set 存在于新生代E中,用于保存新生代对象对应的脏卡
    • 脏卡:O被划分为多个区域(一个区域512K),如果该区域引用了新生代对象,则该区域被称为脏卡(直接到脏卡中遍历寻找对应新生代的GC Root)
  • 在脏卡引用变更时通过post-write barried + dirty card queue 更新(异步操作)
  • concurrent refinement threads 更新 Remembered Set

img

(9). Remark重新标记阶段

在垃圾回收时,收集器处理对象的过程中

黑色:已被处理,需要保留的

灰色:正在处理中的

白色:还未处理的

img

但是在并发标记过程中,有可能A被处理了以后未引用C,但该处理过程还未结束,在处理过程结束之前A引用了C,这时就会用到remark

过程如下

  • 之前C未被引用,这时A引用了C,就会给C加一个写屏障,写屏障的指令会被执行,将C放入一个队列当中,并将C变为 处理中 状态
  • 并发标记阶段结束以后,重新标记阶段会STW,然后将放在该队列中的对象重新处理,发现有强引用引用它,就会处理它

img

以下是G1的后续优化内容

(10). JDK 8u20 字符串去重

例子

String s1 = new String("hello"); // char[]{'h','e','l','l','o'}
String s2 = new String("hello"); // char[]{'h','e','l','l','o'}

过程

  • 将所有新分配的字符串(底层是char[])放入一个队列
  • 当新生代回收时,G1并发检查队列中是否有重复的字符串
  • 如果字符串的值一样,就让他们引用同一个字符串对象
  • 注意,其与String.intern()的区别
    • intern关注的是字符串对象
    • 字符串去重关注的是char[]
    • 两者在JVM内部,使用了不同的字符串表

优点与缺点

  • 节省了大量内存
  • 新生代回收时间略微增加,导致略微多占用CPU

如果想要使用字符串去重功能需要通过命令打开开关

-XX:+UseStringDeduplication //开启字符串去重
(11). JDK 8u40 并发标记类卸载

在并发标记阶段结束以后,就能知道哪些类不再被使用。如果一个类加载器的所有类都不在使用,则卸载它所加载的所有类

-XX:+ClassUnloadingWithConcurrentMark 默认启用

(12). JDK 8u60 回收巨型对象
  • 一个对象大于region的一半时,就称为巨型对象
  • G1不会对巨型对象进行拷贝
  • 回收时被优先考虑
  • G1会跟踪老年代所有incoming引用,如果老年代incoming引用为0的巨型对象就可以在新生代垃圾回收时处理掉

img

(13). JDK 9 并发标记起始时间的调整:
  • 并发标记必须在堆空间占满前完成,否则退化为 FullGC
  • JDK 9 之前需要使用 -XX:InitiatingHeapOccupancyPercent
  • JDK 9 可以动态调整
    • ​ -XX:InitiatingHeapOccupancyPercent 用来设置初始值
    • ​ 进行数据采样并动态调整
    • ​ 总会添加一个安全的空档空间

5、GC 调优(需要结合书再看)

查看虚拟机参数命令; 在idea的终端中输入

"D:\Program Files\Java\jdk1.8.0_221\bin\java" -XX:+PrintFlagsFinal -version | findstr "GC"

可以根据参数去查询具体的信息; 运行结果

D:\Ajvm\out\production\jvm\day2>"D:\Program Files\Java\jdk1.8.0_221\bin\java" -XX:+PrintFlagsFinal -version | findstr "GC"
    uintx AdaptiveSizeMajorGCDecayTimeScale         = 10                                  {product}
    uintx AutoGCSelectPauseMillis                   = 5000                                {product}
     bool BindGCTaskThreadsToCPUs                   = false                               {product}
    uintx CMSFullGCsBeforeCompaction                = 0                                   {product}
    uintx ConcGCThreads                             = 0                                   {product}
     bool DisableExplicitGC                         = false                               {product}
     bool ExplicitGCInvokesConcurrent               = false                               {product}
     bool ExplicitGCInvokesConcurrentAndUnloadsClasses  = false                               {product}
    uintx G1MixedGCCountTarget                      = 8                                   {product}
    uintx GCDrainStackTargetSize                    = 64                                  {product}
    uintx GCHeapFreeLimit                           = 2                                   {product}
    uintx GCLockerEdenExpansionPercent              = 5                                   {product}
     bool GCLockerInvokesConcurrent                 = false                               {product}
    uintx GCLogFileSize                             = 8192                                {product}
    uintx GCPauseIntervalMillis                     = 0                                   {product}
    uintx GCTaskTimeStampEntries                    = 200                                 {product}
    uintx GCTimeLimit                               = 98                                  {product}
    uintx GCTimeRatio                               = 99                                  {product}
     bool HeapDumpAfterFullGC                       = false                               {manageable}
     bool HeapDumpBeforeFullGC                      = false                               {manageable}
    uintx HeapSizePerGCThread                       = 87241520                            {product}
    uintx MaxGCMinorPauseMillis                     = 4294967295                          {product}
    uintx MaxGCPauseMillis                          = 4294967295                          {product}
    uintx NumberOfGCLogFiles                        = 0                                   {product}
     intx ParGCArrayScanChunk                       = 50                                  {product}
    uintx ParGCDesiredObjsFromOverflowList          = 20                                  {product}
     bool ParGCTrimOverflow                         = true                                {product}
     bool ParGCUseLocalOverflow                     = false                               {product}
    uintx ParallelGCBufferWastePct                  = 10                                  {product}
    uintx ParallelGCThreads                         = 13                                  {product}
     bool ParallelGCVerbose                         = false                               {product}
     bool PrintClassHistogramAfterFullGC            = false                               {manageable}
     bool PrintClassHistogramBeforeFullGC           = false                               {manageable}
     bool PrintGC                                   = false                               {manageable}
     bool PrintGCApplicationConcurrentTime          = false                               {product}
     bool PrintGCApplicationStoppedTime             = false                               {product}
     bool PrintGCCause                              = true                                {product}
     bool PrintGCDateStamps                         = false                               {manageable}
     bool PrintGCDetails                            = false                               {manageable}
     bool PrintGCID                                 = false                               {manageable}
     bool PrintGCTaskTimeStamps                     = false                               {product}
     bool PrintGCTimeStamps                         = false                               {manageable}
     bool PrintHeapAtGC                             = false                               {product rw}
     bool PrintHeapAtGCExtended                     = false                               {product rw}
     bool PrintJNIGCStalls                          = false                               {product}
     bool PrintParallelOldGCPhaseTimes              = false                               {product}
     bool PrintReferenceGC                          = false                               {product}
     bool ScavengeBeforeFullGC                      = true                                {product}
     bool TraceDynamicGCThreads                     = false                               {product}
     bool TraceParallelOldGCTasks                   = false                               {product}
     bool UseAdaptiveGCBoundary                     = false                               {product}
     bool UseAdaptiveSizeDecayMajorGCCost           = true                                {product}
     bool UseAdaptiveSizePolicyWithSystemGC         = false                               {product}
     bool UseAutoGCSelectPolicy                     = false                               {product}
     bool UseConcMarkSweepGC                        = false                               {product}
     bool UseDynamicNumberOfGCThreads               = false                               {product}
     bool UseG1GC                                   = false                               {product}
     bool UseGCLogFileRotation                      = false                               {product}
     bool UseGCOverheadLimit                        = true                                {product}
     bool UseGCTaskAffinity                         = false                               {product}
     bool UseMaximumCompactionOnSystemGC            = true                                {product}
     bool UseParNewGC                               = false                               {product}
     bool UseParallelGC                            := true                                {product}
     bool UseParallelOldGC                          = true                                {product}
     bool UseSerialGC                               = false                               {product}
java version "1.8.0_221"
Java(TM) SE Runtime Environment (build 1.8.0_221-b11)
Java HotSpot(TM) 64-Bit Server VM (build 25.221-b11, mixed mode)

调优领域
  • 内存
  • 锁竞争
  • CPU占用
  • IO
  • GC
确定目标

低延迟/高吞吐量? 选择合适的垃圾回收器

  • CMS G1 ZGC
  • ParallelGC
  • Zing
最快的GC是不发生GC

首先排除减少因为自身编写的代码而引发的内存问题

  • 查看Full GC前后的内存占用,考虑以下几个问题
    • 数据是不是太多?
    • 数据表示是否太臃肿
      • 对象图
      • 对象大小
    • 是否存在内存泄漏
新生代调优
  • 新生代的特点

    • 所有的new操作分配内存都是非常廉价的
      • TLAB
    • 死亡对象回收零代价
    • 大部分对象用过即死(朝生夕死)
    • MInor GC 所用时间远小于Full GC
  • 新生代内存越大越好么?

    • 不是

      • 新生代内存太小:频繁触发Minor GC,会STW,会使得吞吐量下降
      • 新生代内存太大:老年代内存占比有所降低,会更频繁地触发Full GC。而且触发Minor GC时,清理新生代所花费的时间会更长
    • 新生代内存设置为能容纳 [并发量*(请求-响应)] 的数据为宜

幸存区调优
  • 幸存区需要能够保存 当前活跃对象+需要晋升的对象

  • 晋升阈值配置得当,让长时间存活的对象尽快晋升

老年代调优

以 CMS 为例 :

  • CMS 的老年代内存越大越好
  • 先尝试不做调优,如果没有 Full GC 那么已经…,否则先尝试调优新生代
  • 观察发生 Full GC 时老年代内存占用,将老年代内存预设调大 1/4 ~ 1/3
    • -XX:CMSInitiatingOccupancyFraction=percent

四、类加载与字节码技术

img

1、类文件结构 (以下所有内容了解即可)

// HelloWorld 示例
public class HelloWorld 
{
    public static void main(String[] args) 
    {
        System.out.println("hello world");
    }
}

执行 javac -parameters -d . HellowWorld.java

编译为 HelloWorld.class 得到的字节码文件是这个样子的:

0000000 ca fe ba be 00 00 00 34 00 23 0a 00 06 00 15 09 
0000020 00 16 00 17 08 00 18 0a 00 19 00 1a 07 00 1b 07 
0000040 00 1c 01 00 06 3c 69 6e 69 74 3e 01 00 03 28 29 
0000060 56 01 00 04 43 6f 64 65 01 00 0f 4c 69 6e 65 4e 
0000100 75 6d 62 65 72 54 61 62 6c 65 01 00 12 4c 6f 63 
0000120 61 6c 56 61 72 69 61 62 6c 65 54 61 62 6c 65 01 
0000140 00 04 74 68 69 73 01 00 1d 4c 63 6e 2f 69 74 63 
0000160 61 73 74 2f 6a 76 6d 2f 74 35 2f 48 65 6c 6c 6f 
0000200 57 6f 72 6c 64 3b 01 00 04 6d 61 69 6e 01 00 16 
0000220 28 5b 4c 6a 61 76 61 2f 6c 61 6e 67 2f 53 74 72 
0000240 69 6e 67 3b 29 56 01 00 04 61 72 67 73 01 00 13 
0000260 5b 4c 6a 61 76 61 2f 6c 61 6e 67 2f 53 74 72 69 
0000300 6e 67 3b 01 00 10 4d 65 74 68 6f 64 50 61 72 61 
0000320 6d 65 74 65 72 73 01 00 0a 53 6f 75 72 63 65 46 
0000340 69 6c 65 01 00 0f 48 65 6c 6c 6f 57 6f 72 6c 64
0000360 2e 6a 61 76 61 0c 00 07 00 08 07 00 1d 0c 00 1e 
0000400 00 1f 01 00 0b 68 65 6c 6c 6f 20 77 6f 72 6c 64 
0000420 07 00 20 0c 00 21 00 22 01 00 1b 63 6e 2f 69 74 
0000440 63 61 73 74 2f 6a 76 6d 2f 74 35 2f 48 65 6c 6c 
0000460 6f 57 6f 72 6c 64 01 00 10 6a 61 76 61 2f 6c 61 
0000500 6e 67 2f 4f 62 6a 65 63 74 01 00 10 6a 61 76 61 
0000520 2f 6c 61 6e 67 2f 53 79 73 74 65 6d 01 00 03 6f 
0000540 75 74 01 00 15 4c 6a 61 76 61 2f 69 6f 2f 50 72 
0000560 69 6e 74 53 74 72 65 61 6d 3b 01 00 13 6a 61 76 
0000600 61 2f 69 6f 2f 50 72 69 6e 74 53 74 72 65 61 6d 
0000620 01 00 07 70 72 69 6e 74 6c 6e 01 00 15 28 4c 6a 
0000640 61 76 61 2f 6c 61 6e 67 2f 53 74 72 69 6e 67 3b 
0000660 29 56 00 21 00 05 00 06 00 00 00 00 00 02 00 01 
0000700 00 07 00 08 00 01 00 09 00 00 00 2f 00 01 00 01 
0000720 00 00 00 05 2a b7 00 01 b1 00 00 00 02 00 0a 00 
0000740 00 00 06 00 01 00 00 00 04 00 0b 00 00 00 0c 00 
0000760 01 00 00 00 05 00 0c 00 0d 00 00 00 09 00 0e 00 
0001000 0f 00 02 00 09 00 00 00 37 00 02 00 01 00 00 00 
0001020 09 b2 00 02 12 03 b6 00 04 b1 00 00 00 02 00 0a 
0001040 00 00 00 0a 00 02 00 00 00 06 00 08 00 07 00 0b 
0001060 00 00 00 0c 00 01 00 00 00 09 00 10 00 11 00 00 
0001100 00 12 00 00 00 05 01 00 10 00 00 00 01 00 13 00 
0001120 00 00 02 00 14

根据 JVM 规范,类文件结构如下

classFile
{
	u4 			   magic //魔数
    u2             minor_version;   //小版本号  
    u2             major_version;   //主版本号 
    u2             constant_pool_count;  //常量池  
    cp_info        constant_pool[constant_pool_count-1];    
    u2             access_flags;    //访问修饰
    u2             this_class;    	//类的包名、类名
    u2             super_class;   	//父类的信息
    u2             interfaces_count;    //接口的信息
    u2             interfaces[interfaces_count];   
    u2             fields_count;    //类中的变量信息
    field_info     fields[fields_count];   
    u2             methods_count;    //类中的方法信息
    method_info    methods[methods_count];    
    u2             attributes_count;    //类中附加属性信息
    attribute_info attributes[attributes_count];
}
1.1. 魔数

u4 magic

对应字节码文件 0~3 字节,表示它是否是【class】类型的文件

0000000 ca fe ba be 00 00 00 34 00 23 0a 00 06 00 15 09

1.2. 版本

u2 minor_version; u2 major_version;

对应字节码文件 4~7 字节,表示类的版本 00 34(52) 表示是 Java 8

0000000 ca fe ba be 00 00 00 34 00 23 0a 00 06 00 15 09

34H = 52,代表JDK8

1.3. 常量池

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-k1GmzcTA-1669108012089)(C:\Users\vanpersie\AppData\Roaming\Typora\typora-user-images\image-20221118130848564.png)]

8-9个字节,表示常量池长度,00 23(转化为十六进制后是35)表示常量池有 #1~#34项,注意 #0 项不计入,也没有值

0000000 ca fe ba be 00 00 00 34 00 23 0a 00 06 00 15 09

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-qSvi7Dp6-1669108012089)(C:\Users\vanpersie\AppData\Roaming\Typora\typora-user-images\image-20221118133609787.png)]

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-zj8epi7p-1669108012090)(C:\Users\vanpersie\AppData\Roaming\Typora\typora-user-images\image-20221118134315309.png)]

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-oFw69aAP-1669108012090)(C:\Users\vanpersie\AppData\Roaming\Typora\typora-user-images\image-20221118133756609.png)]

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-KHFSpiVf-1669108012090)(C:\Users\vanpersie\AppData\Roaming\Typora\typora-user-images\image-20221118133532730.png)]

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-1XrjXyaz-1669108012090)(C:\Users\vanpersie\AppData\Roaming\Typora\typora-user-images\image-20221118133738387.png)]

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-9BpQnvFt-1669108012091)(C:\Users\vanpersie\AppData\Roaming\Typora\typora-user-images\image-20221118133729537.png)]

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-BSXGIe5H-1669108012091)(C:\Users\vanpersie\AppData\Roaming\Typora\typora-user-images\image-20221118135229665.png)]

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-vPajIUPA-1669108012091)(C:\Users\vanpersie\AppData\Roaming\Typora\typora-user-images\image-20221118135242245.png)]

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-cu0BEhsZ-1669108012091)(C:\Users\vanpersie\AppData\Roaming\Typora\typora-user-images\image-20221118133643209.png)]

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-wsAnrVs3-1669108012092)(C:\Users\vanpersie\AppData\Roaming\Typora\typora-user-images\image-20221118134839036.png)]

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-0gdROtpE-1669108012092)(C:\Users\vanpersie\AppData\Roaming\Typora\typora-user-images\image-20221118134940409.png)]

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-y4U3FiYY-1669108012092)(C:\Users\vanpersie\AppData\Roaming\Typora\typora-user-images\image-20221118134958381.png)]

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-4BmIsKh8-1669108012092)(C:\Users\vanpersie\AppData\Roaming\Typora\typora-user-images\image-20221118135129488.png)]

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-qFlXou4c-1669108012092)(C:\Users\vanpersie\AppData\Roaming\Typora\typora-user-images\image-20221118135113254.png)]

1.4. 访问标识与继承信息

21表示该class的一个类,公共的

0000660 29 56 00 21 00 05 00 06 00 00 00 00 00 02 00 01

05表示根据常量池中 #5 找到本类全限定名

0000660 29 56 00 21 00 05 00 06 00 00 00 00 00 02 00 01

06表示根据常量池中 #6 找到父类全限定名

0000660 29 56 00 21 00 05 00 06 00 00 00 00 00 02 00 01

表示接口的数量,本类为0

0000660 29 56 00 21 00 05 00 06 00 00 00 00 00 02 00 01

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-ZchsLibK-1669108012093)(C:\Users\vanpersie\AppData\Roaming\Typora\typora-user-images\image-20221118160528538.png)]

1.5. Field信息

表示成员变量数量,本类为0

0000660 29 56 00 21 00 05 00 06 00 00 00 00 00 02 00 01

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-FsQ6IxhX-1669108012093)(C:\Users\vanpersie\AppData\Roaming\Typora\typora-user-images\image-20221118161106311.png)]

1.6. Method信息

表示方法数量,本类为2

0000660 29 56 00 21 00 05 00 06 00 00 00 00 00 02 00 01

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-Rx26nTdB-1669108012093)(C:\Users\vanpersie\AppData\Roaming\Typora\typora-user-images\image-20221118161420778.png)]

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-pn1IwTU0-1669108012094)(C:\Users\vanpersie\AppData\Roaming\Typora\typora-user-images\image-20221118161454453.png)]

1.7. 附加属性

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-aGrxB3Gb-1669108012094)(C:\Users\vanpersie\AppData\Roaming\Typora\typora-user-images\image-20221119133007427.png)]

2、字节码指令

2.1. 入门 (了解即可)

public cn.ithcast.jvm.t5.HelloWorld(); 构造方法的字节码指令

2a b7 00 01 b1
  1. 2a=> aload_0 加载slot 0 的局部变量,即this,做为下面的 invokespecial 构造方法调用的参数
  2. b7=> invokespecial 预备调用构造方法,哪个方法呢?
  3. 00 01 引用常量池中 #1 项, 即【Method java/lang/Object.“”😦)V】
  4. b1表示返回

另一个是public static void main(java.lang.String[]); 主方法的字节码指令

b2 00 02 12 03 b6 00 04 b1
  1. b2 => getstatic 用来加载静态变量,哪个静态变量呢?
  2. 00 02 引用常量池中 #2 项, 即【Field java/lang/System.out:Ljava/io/PrintStream;】
  3. 12 => ldc 加载参数,哪个参数呢?
  4. 03 引用常量池中 #3项,即【String hello world】
  5. b6 => invokevirtual 预备调用成员方法,哪个方法呢?
  6. 00 04 引用常量池中 #4 项,即【Method java/io/PrintStream.println:(Ljava/lang/String;)V】
  7. b1 表示返回

可参考

https://docs.oracle.com/javase/specs/jvms/se8/html/jvms-6.html#jvms-6.5

2.2. javap工具

Oracle 提供了 javap 工具来反编译 class 文件

javap -v 类名.class
D:\Ajvm\out\production\jvm\day3>javap -v HelloWorld.class
Classfile /D:/Ajvm/out/production/jvm/day3/HelloWorld.class
  Last modified 2022-11-18; size 543 bytes
  MD5 checksum c71e1a5d7846d6ee58b3b75ad31a1374
  Compiled from "HelloWorld.java"
public class day3.HelloWorld
  minor version: 0
  major version: 52
  flags: ACC_PUBLIC, ACC_SUPER //访问修饰
Constant pool: //常量池
   #1 = Methodref          #6.#20         // java/lang/Object."<init>":()V
   #2 = Fieldref           #21.#22        // java/lang/System.out:Ljava/io/PrintStream;
   #3 = String             #23            // hello world
   #4 = Methodref          #24.#25        // java/io/PrintStream.println:(Ljava/lang/String;)V
   #5 = Class              #26            // day3/HelloWorld
   #6 = Class              #27            // java/lang/Object
   #7 = Utf8               <init>
   #8 = Utf8               ()V
   #9 = Utf8               Code
  #10 = Utf8               LineNumberTable
  #11 = Utf8               LocalVariableTable
  #12 = Utf8               this
  #13 = Utf8               Lday3/HelloWorld;
  #14 = Utf8               main
  #15 = Utf8               ([Ljava/lang/String;)V
  #16 = Utf8               args
  #17 = Utf8               [Ljava/lang/String;
  #18 = Utf8               SourceFile
  #19 = Utf8               HelloWorld.java
  #20 = NameAndType        #7:#8          // "<init>":()V
  #21 = Class              #28            // java/lang/System
  #22 = NameAndType        #29:#30        // out:Ljava/io/PrintStream;
  #23 = Utf8               hello world
  #24 = Class              #31            // java/io/PrintStream
  #25 = NameAndType        #32:#33        // println:(Ljava/lang/String;)V
  #26 = Utf8               day3/HelloWorld
  #27 = Utf8               java/lang/Object
  #28 = Utf8               java/lang/System
  #29 = Utf8               out
  #30 = Utf8               Ljava/io/PrintStream;
  #31 = Utf8               java/io/PrintStream
  #32 = Utf8               println
  #33 = Utf8               (Ljava/lang/String;)V
{
  public day3.HelloWorld(); //构造方法部分
    descriptor: ()V
    flags: ACC_PUBLIC
    Code:
      stack=1, locals=1, args_size=1
         0: aload_0							//0\1\4代表字节码中的行号
         1: invokespecial #1                  // Method java/lang/Object."<init>":()V
         4: return
      LineNumberTable:
        line 3: 0		//line 3 表示java代码中的行号 : 0 表示字节码中的行号
      LocalVariableTable:	//局部变量表
        Start  Length  Slot  Name   Signature
            0       5     0  this   Lday3/HelloWorld;

  public static void main(java.lang.String[]); //主方法部分
    descriptor: ([Ljava/lang/String;)V //参数类型
    flags: ACC_PUBLIC, ACC_STATIC	//访问修饰类型
    Code:
      stack=2, locals=1, args_size=1
         0: getstatic     #2                  // Field java/lang/System.out:Ljava/io/PrintStream;
         3: ldc           #3                  // String hello world
         5: invokevirtual #4                  // Method java/io/PrintStream.println:(Ljava/lang/String;)V
         8: return
      LineNumberTable:
        line 7: 0
        line 8: 8
      LocalVariableTable:
        Start  Length  Slot  Name   Signature
            0       9     0  args   [Ljava/lang/String;
}
SourceFile: "HelloWorld.java"
2.3. 图解方法执行流程
(1) 演示代码
public class Demo3_1 
{    
	public static void main(String[] args) 
    {        
		int a = 10;        
		int b = Short.MAX_VALUE + 1;        
		int c = a + b;        
		System.out.println(c);   
    } 
}
(2) 对应的字节码文件
D:\Ajvm\out\production\jvm\day3>javap -v Demo3_1.class
Classfile /D:/Ajvm/out/production/jvm/day3/Demo3_1.class
  Last modified 2022-11-19; size 593 bytes
  MD5 checksum 5ba1e77b0b16b3e604ed603bff47114e
  Compiled from "Demo3_1.java"
public class day3.Demo3_1
  minor version: 0
  major version: 52
  flags: ACC_PUBLIC, ACC_SUPER
Constant pool:
   #1 = Methodref          #7.#25         // java/lang/Object."<init>":()V
   #2 = Class              #26            // java/lang/Short
   #3 = Integer            32768
   #4 = Fieldref           #27.#28        // java/lang/System.out:Ljava/io/PrintStream;
   #5 = Methodref          #29.#30        // java/io/PrintStream.println:(I)V
   #6 = Class              #31            // day3/Demo3_1
   #7 = Class              #32            // java/lang/Object
   #8 = Utf8               <init>
   #9 = Utf8               ()V
  #10 = Utf8               Code
  #11 = Utf8               LineNumberTable
  #12 = Utf8               LocalVariableTable
  #13 = Utf8               this
  #14 = Utf8               Lday3/Demo3_1;
  #15 = Utf8               main
  #16 = Utf8               ([Ljava/lang/String;)V
  #17 = Utf8               args
  #18 = Utf8               [Ljava/lang/String;
  #19 = Utf8               a
  #20 = Utf8               I
  #21 = Utf8               b
  #22 = Utf8               c
  #23 = Utf8               SourceFile
  #24 = Utf8               Demo3_1.java
  #25 = NameAndType        #8:#9          // "<init>":()V
  #26 = Utf8               java/lang/Short
  #27 = Class              #33            // java/lang/System
  #28 = NameAndType        #34:#35        // out:Ljava/io/PrintStream;
  #29 = Class              #36            // java/io/PrintStream
  #30 = NameAndType        #37:#38        // println:(I)V
  #31 = Utf8               day3/Demo3_1
  #32 = Utf8               java/lang/Object
  #33 = Utf8               java/lang/System
  #34 = Utf8               out
  #35 = Utf8               Ljava/io/PrintStream;
  #36 = Utf8               java/io/PrintStream
  #37 = Utf8               println
  #38 = Utf8               (I)V
{
  public day3.Demo3_1();
    descriptor: ()V
    flags: ACC_PUBLIC
    Code:
      stack=1, locals=1, args_size=1
         0: aload_0
         1: invokespecial #1                  // Method java/lang/Object."<init>":()V
         4: return
      LineNumberTable:
        line 3: 0
      LocalVariableTable:
        Start  Length  Slot  Name   Signature
            0       5     0  this   Lday3/Demo3_1;

  public static void main(java.lang.String[]);
    descriptor: ([Ljava/lang/String;)V
    flags: ACC_PUBLIC, ACC_STATIC
    Code:
      stack=2, locals=4, args_size=1
         0: bipush        10
         2: istore_1
         3: ldc           #3                  // int 32768
         5: istore_2
         6: iload_1
         7: iload_2
         8: iadd
         9: istore_3
        10: getstatic     #4                  // Field java/lang/System.out:Ljava/io/PrintStream;
        13: iload_3
        14: invokevirtual #5                  // Method java/io/PrintStream.println:(I)V
        17: return
      LineNumberTable:
        line 7: 0
        line 8: 3
        line 9: 6
        line 10: 10
        line 11: 17
      LocalVariableTable:
        Start  Length  Slot  Name   Signature
            0      18     0  args   [Ljava/lang/String;
            3      15     1     a   I
            6      12     2     b   I
           10       8     3     c   I
}
SourceFile: "Demo3_1.java"
(3) 常量池载入运行时常量池

常量池也属于方法区,只不过这里单独提出来观察

img

(4) 方法字节码载入方法区

img

(5) main 线程开始运行,分配栈帧内存

(stack=2 操作数栈的深度,locals=4 局部变量表的长度)

对应操作数栈(蓝色的)有2个空间(每个空间4个字节),局部变量表 (绿色的) 中有4个槽位

img

(6) 执行引擎开始执行字节码

bipush 10

  • 将一个 byte 压入操作数栈(其长度会补齐 4 个字节),类似的指令还有

    • sipush 将一个 short 压入操作数栈(其长度会补齐 4 个字节)
    • ldc 将一个 int 压入操作数栈
    • ldc2_w 将一个 long 压入操作数栈(分两次压入,因为 long 是 8 个字节)
    • 这里小的数字都是和字节码指令存在一起,超过 short 范围的数字存入了常量池

img

istore 1

将操作数栈栈顶元素弹出,放入局部变量表的slot 1 (槽位)中

对应代码中的

a = 10

img

img

ldc #3

读取运行时常量池中#3,即32768(超过short最大值范围的数会被放到运行时常量池中),将其加载到操作数栈中

注意 Short.MAX_VALUE 是 32767,所以 32768 = Short.MAX_VALUE + 1 实际是在编译期间计算好的

img

istore 2

将操作数栈中的元素弹出,放到局部变量表的2号位置

img

img

iload1 iload2

将局部变量表中1号位置和2号位置的元素放入操作数栈中

  • 因为只能在操作数栈中执行运算操作

img

img

iadd

将操作数栈中的两个元素弹出栈并相加,结果再压入操作数栈中

img

img

istore 3

将操作数栈中的元素弹出,放入局部变量表的3号位置

img

img

getstatic #4

在运行时常量池中找到#4,发现是一个对象

在堆内存中找到该对象,并将其引用放入操作数栈中

img

img

iload 3

将局部变量表中3号位置的元素压入操作数栈中

img

invokevirtual 5

找到常量池 #5 项,定位到方法区 java/io/PrintStream.println:(I)V 方法

生成新的栈帧(分配 locals、stack等)

传递参数,执行新栈帧中的字节码

img

执行完毕,弹出栈帧

清除 main 操作数栈内容

img

return
完成 main 方法调用,弹出 main 栈帧,程序结束

2.4. 练习,a++问题

代码

public class Demo1
{
    public static void main(String[] args)
    {
        int a = 10;
        int b = a++ + ++a + a--;
        System.out.println(a);
        System.out.println(b);
    }
}

对应的字节码文件

D:\Ajvm\out\production\jvm\day3>javap -v Demo1.class
Classfile /D:/Ajvm/out/production/jvm/day3/Demo1.class
  Last modified 2022-11-19; size 562 bytes
  MD5 checksum 292197a602a195e776bc92cba4dc942d
  Compiled from "Demo1.java"
public class day3.Demo1
  minor version: 0
  major version: 52
  flags: ACC_PUBLIC, ACC_SUPER
Constant pool:
   #1 = Methodref          #5.#22         // java/lang/Object."<init>":()V
   #2 = Fieldref           #23.#24        // java/lang/System.out:Ljava/io/PrintStream;
   #3 = Methodref          #25.#26        // java/io/PrintStream.println:(I)V
   #4 = Class              #27            // day3/Demo1
   #5 = Class              #28            // java/lang/Object
   #6 = Utf8               <init>
   #7 = Utf8               ()V
   #8 = Utf8               Code
   #9 = Utf8               LineNumberTable
  #10 = Utf8               LocalVariableTable
  #11 = Utf8               this
  #12 = Utf8               Lday3/Demo1;
  #13 = Utf8               main
  #14 = Utf8               ([Ljava/lang/String;)V
  #15 = Utf8               args
  #16 = Utf8               [Ljava/lang/String;
  #17 = Utf8               a
  #18 = Utf8               I
  #19 = Utf8               b
  #20 = Utf8               SourceFile
  #21 = Utf8               Demo1.java
  #22 = NameAndType        #6:#7          // "<init>":()V
  #23 = Class              #29            // java/lang/System
  #24 = NameAndType        #30:#31        // out:Ljava/io/PrintStream;
  #25 = Class              #32            // java/io/PrintStream
  #26 = NameAndType        #33:#34        // println:(I)V
  #27 = Utf8               day3/Demo1
  #28 = Utf8               java/lang/Object
  #29 = Utf8               java/lang/System
  #30 = Utf8               out
  #31 = Utf8               Ljava/io/PrintStream;
  #32 = Utf8               java/io/PrintStream
  #33 = Utf8               println
  #34 = Utf8               (I)V
{
  public day3.Demo1();
    descriptor: ()V
    flags: ACC_PUBLIC
    Code:
      stack=1, locals=1, args_size=1
         0: aload_0
         1: invokespecial #1                  // Method java/lang/Object."<init>":()V
         4: return
      LineNumberTable:
        line 3: 0
      LocalVariableTable:
        Start  Length  Slot  Name   Signature
            0       5     0  this   Lday3/Demo1;

  public static void main(java.lang.String[]);
    descriptor: ([Ljava/lang/String;)V
    flags: ACC_PUBLIC, ACC_STATIC
    Code:
      stack=2, locals=3, args_size=1
         0: bipush        10  //通过图解主要分析 这部分开始 的内容
         2: istore_1
         3: iload_1
         4: iinc          1, 1
         7: iinc          1, 1
        10: iload_1
        11: iadd
        12: iload_1
        13: iinc          1, -1
        16: iadd
        17: istore_2     		//通过图解主要分析 这部分结束 的内容
        18: getstatic     #2                  // Field java/lang/System.out:Ljava/io/PrintStream;
        21: iload_1
        22: invokevirtual #3                  // Method java/io/PrintStream.println:(I)V
        25: getstatic     #2                  // Field java/lang/System.out:Ljava/io/PrintStream;
        28: iload_2
        29: invokevirtual #3                  // Method java/io/PrintStream.println:(I)V
        32: return
      LineNumberTable:
        line 7: 0
        line 8: 3
        line 9: 18
        line 10: 25
        line 11: 32
      LocalVariableTable:
        Start  Length  Slot  Name   Signature
            0      33     0  args   [Ljava/lang/String;
            3      30     1     a   I
           18      15     2     b   I
}
SourceFile: "Demo1.java"

通过图标分析:

  • 执行自增的指令叫iinc;但注意iinc指令是直接在局部变量表slot上进行运算,不在操作数栈上运算
  • a++ 先执行 iload, 再执行 iinc
  • ++a 先执行iinc, 再执行iload

(1)bipush 10

​ 把 10 放入到操作数栈中

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-tMjqhD8c-1669108012099)(C:\Users\vanpersie\AppData\Roaming\Typora\typora-user-images\image-20221119145606772.png)]

(2)istore1

​ 把 10 放到 slot 为1 的局部变量表中

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-XLVelNaO-1669108012099)(C:\Users\vanpersie\AppData\Roaming\Typora\typora-user-images\image-20221119145923951.png)]

​ (3) iload 1

​ 把 slot 1 的元素放入到操作数栈中

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-XunYOpoa-1669108012099)(C:\Users\vanpersie\AppData\Roaming\Typora\typora-user-images\image-20221119150151093.png)]

(4)iinc 1,1

​ 对slot 1 的元素进行自增1

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-k2jpZnQP-1669108012099)(C:\Users\vanpersie\AppData\Roaming\Typora\typora-user-images\image-20221119150346541.png)]

(5)iinc 1,1

​ 对slot 1 的元素进行自增1

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-IYWUyQHT-1669108012099)(C:\Users\vanpersie\AppData\Roaming\Typora\typora-user-images\image-20221119150441093.png)]

​ (6) iload 1

​ 把 slot 1 的元素放入到操作数栈中

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-PVLtyE9c-1669108012100)(C:\Users\vanpersie\AppData\Roaming\Typora\typora-user-images\image-20221119150508338.png)]

​ (7)iadd

​ 将操作数栈中的元素先弹出相加,再把结果压回到操作数栈中

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-OOEiKp3g-1669108012103)(C:\Users\vanpersie\AppData\Roaming\Typora\typora-user-images\image-20221119150611104.png)]

​ (8)iload 1

​ 把 slot 1 的元素放入到操作数栈中

​ [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-2ecqarkH-1669108012104)(C:\Users\vanpersie\AppData\Roaming\Typora\typora-user-images\image-20221119150656166.png)]

(9)iinc 1,-1

​ 对 slot 1 的元素进行自增-1

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-PweP2Be2-1669108012104)(C:\Users\vanpersie\AppData\Roaming\Typora\typora-user-images\image-20221119150804521.png)]

(10)iadd

​ 将操作数栈中的元素先弹出相加,再把结果压回到操作数栈中

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-n2UYyL6G-1669108012105)(C:\Users\vanpersie\AppData\Roaming\Typora\typora-user-images\image-20221119150848730.png)]

​ (11) istore 2

​ 将34 存入到局部变量表 slot 2 的位置中

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-73sCqEiz-1669108012105)(C:\Users\vanpersie\AppData\Roaming\Typora\typora-user-images\image-20221119150959263.png)]

2.5.条件判断指令

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-VOPmo02k-1669108012105)(C:\Users\vanpersie\AppData\Roaming\Typora\typora-user-images\image-20221119151144562.png)]

几点说明:

  • byte, short, char都会按 int 比较,因为操作数栈都是4个字节
  • goto 用来进行跳转到指定行号的字节码

源码

public class Demo3_3
{
    public static void main(String[] args) 
    {
        int a = 0;
        if(a == 0) 
        {
            a = 10;
        }
        else 
        {
            a = 20;
        }
    }
}

字节码

 0: iconst_0  	//得到0的常量,放入操作数栈
 1: istore_1	//放入到slot 1的局部变量表中
 2: iload_1		//把slot 1的元素放入到操作数栈中
 3: ifne          12 	//判断是否 !=0,如果不等于0,会跳到12行;等于0继续执行下一行
 6: bipush        10	//把10放入到操作数栈中
 8: istore_1	//放入到slot 1的局部变量表中
 9: goto          15	//跳转到15行
 12: bipush        20	//把20放入到操作数栈中
 14: istore_1	//放入到slot 1的局部变量表中
 15: return		//返回
2.6. 循环控制指令

其实循环控制指令还是前面介绍的那些指令,例如while循环

代码

public class Demo3_4
{
    public static void main(String[] args)
    {
        int a = 0;
        while (a < 10)
        {
            a++;
        }
    }
}

字节码

0: iconst_0		//得到0的常量,放入操作数栈
1: istore_1		//放入到slot 1的局部变量表中
2: iload_1		//把slot 1的元素放入到操作数栈中
3: bipush        10		//把10放入到操作数栈中
5: if_icmpge     14		//判断0 是否 大于等于 14;如果大于,跳转到14行;否则继续执行下一行
8: iinc          1, 1	//对slot 1的元素执行自增操作1的操作
11: goto          2		//跳转到2行
14: return		//返回
2.7. 练习,判断结果

代码

public class Demo2 
{
	public static void main(String[] args) 
    {
		int i=0;
		int x=0;
		while(i<10) 
        {
			x = x++; //x++返回的是++之前的值,所以x始终为0
			i++;
		}
		System.out.println(x); //结果为0
	}
}

为什么最终的x结果为0呢? 通过分析字节码指令即可知晓

Code:
     stack=2, locals=3, args_size=1	//操作数栈分配2个空间,局部变量表分配3个空间
        0: iconst_0		//准备一个常数0
        1: istore_1		//将常数0放入局部变量表的1号槽位 i=0
        2: iconst_0		//准备一个常数0
        3: istore_2		//将常数0放入局部变量的2号槽位 x=0	
        4: iload_1		//将局部变量表1号槽位的数放入操作数栈中
        5: bipush        10	//将数字10放入操作数栈中,此时操作数栈中有2个数
        7: if_icmpge     21	//比较操作数栈中的两个数,如果i >= 10,就跳转到21。否则继续执行下一行
       10: iload_2		//将局部变量2号槽位的数放入操作数栈中,放入的值是0
       11: iinc          2, 1	//将局部变量2号槽位的数加1,自增后,槽位中的值为1
       14: istore_2		//将操作数栈中的数放入到局部变量表的2号槽位,2号槽位的值又变为了0
       15: iinc          1, 1 //1号槽位的值自增1
       18: goto          4 //跳转到第4条指令
       21: getstatic     #2                  // Field java/lang/System.out:Ljava/io/PrintStream;
       24: iload_2
       25: invokevirtual #3                  // Method java/io/PrintStream.println:(I)V
       28: return
2.8. 构造方法
2.8.1. cinit()V
public class Demo3 
{
	static int i = 10;

	static 
	{
		i = 20;
	}

	static 
	{
		i = 30;
	}

	public static void main(String[] args) 
	{
		System.out.println(i); //结果为30
	}
}

编译器会按从上至下的顺序,收集所有 static 静态代码块和静态成员赋值的代码,合并为一个特殊的方法 ()V :

stack=1, locals=0, args_size=0
    0: bipush        10
    2: putstatic     #3                  // Field i:I
    5: bipush        20
    7: putstatic     #3                  // Field i:I
    10: bipush        30
    12: putstatic     #3                  // Field i:I
    15: return
2.8.2 init()V
public class Demo4 
{
	private String a = "s1";
    
	{
		b = 20;
	}

	private int b = 10;
    
	{
		a = "s2";
	}

	public Demo4(String a, int b) 
   	{
		this.a = a;
		this.b = b;
	}

	public static void main(String[] args) 
    {
		Demo4 d = new Demo4("s3", 30);
		System.out.println(d.a); //s3
		System.out.println(d.b); //30
	}
}

编译器会按从上至下的顺序,收集所有 {} 代码块和成员变量赋值的代码,形成新的构造方法,但原始构造方法内的代码总是在新的构造方法的最后

Code:
     stack=2, locals=3, args_size=3
        0: aload_0
        1: invokespecial #1                  // Method java/lang/Object."<init>":()V
        4: aload_0
        5: ldc           #2                  // 把 s1 加载到操作数栈
        7: putfield      #3                  // 把 s1 赋值给 a
       10: aload_0
       11: bipush        20					 // 把 20 加载到操作数栈
       13: putfield      #4                  // 把 20 赋值给 b
       16: aload_0
       17: bipush        10					 // 把 10 加载到操作数栈	
       19: putfield      #4                  // 把 10 赋值给 b
       22: aload_0
       23: ldc           #5                  // 把 s2 加载到操作数栈
       25: putfield      #3                  // 把 s2 赋值给 a
       //原始构造方法在最后执行
       28: aload_0
       29: aload_1
       30: putfield      #3                  // 把 s3 赋值给 a
       33: aload_0
       34: iload_2							 // 把slot 2 位置的元素放入到操作数栈中	
       35: putfield      #4                  // 把 30 赋值给 b
       38: return							 //返回

总结:cinit初始化的是类成员变量(static),在类加载时就已经调用了;init是在调用构造方法时调用,对应的是非static的成员变量

2.9. 方法调用
public class Demo5 
{
	public Demo5() {} //构造方法

	private void test1() {} //私有方法

	private final void test2() {} //私有的final方法

	public void test3() {} //公共方法

	public static void test4() {} //静态方法

	public static void main(String[] args) 
    {
		Demo5 demo5 = new Demo5();
		demo5.test1();
		demo5.test2();
		demo5.test3();
        demo5.test(); //通过对象调用静态方法
		Demo5.test4(); //通过类名调用静态方法
	}
}

不同方法在调用时,对应的虚拟机指令有所区别

  • 私有、构造、被final修饰的方法,在调用时都使用invokespecial指令
  • 普通成员方法在调用时,使用invokespecial指令。因为编译期间无法确定该方法的内容,只有在运行期间才能确定
  • 静态方法在调用时使用invokestatic指令
0: new           #2                  // class day3/Demo5
3: dup
4: invokespecial #3                  // Method "<init>":()V
7: astore_1
8: aload_1
9: invokespecial #4                  // Method test1:()V
12: aload_1
13: invokespecial #5                  // Method test2:()V
16: aload_1
17: invokevirtual #6                  // Method test3:()V
20: aload_1
21: pop
22: invokestatic  #7                  // Method test4:()V
25: invokestatic  #7                  // Method test4:()V
28: return
  • new 是创建【对象】,给对象分配堆内存,执行成功会将【对象引用(地址)】压入操作数栈
  • dup 是复制操作数栈栈顶的内容,本例即为【对象引用】。为什么需要两份引用呢,一个是要配合 invokespecial 调用该对象的构造方法 “init”😦)V (会消耗掉栈顶一个引用),另一个要 配合 astore_1 赋值给局部变量
  • 终方法(final),私有方法(private),构造方法都是由 invokespecial 指令来调用,属于静态绑定
  • 普通成员方法是由 invokevirtual 调用,属于动态绑定,即支持多态 成员方法与静态方法调用的另一个区别是,执行方法前是否需要【对象引用】
2.10. 多态原理

代码

/**
 * 演示多态原理,注意加上下面的 JVM 参数,禁用指针压缩
 * -XX:-UseCompressedOops -XX:-UseCompressedClassPointers
 */
public class Demo3_10
{
    public static void test(Animal animal)
    {
        animal.eat();
        System.out.println(animal.toString());
    }

    public static void main(String[] args) throws IOException
    {
        test(new Cat());
        test(new Dog());
        System.in.read();
    }
}

abstract class Animal
{
    public abstract void eat();

    @Override
    public String toString()
    {
        return "我是" + this.getClass().getSimpleName();
    }
}

class Dog extends Animal
{
    @Override
    public void eat()
    {
        System.out.println("啃骨头");
    }
}

class Cat extends Animal
{
    @Override
    public void eat()
    {
        System.out.println("吃鱼");
    }
}
(1) 运行代码

停在System.in.read()方法上,这时运行jps获取进程id

(2) 运行HSDB工具

进入JDK安装目录,执行

java -cp ./lib/sa-jdi.jar sun.jvm.hotspot.HSDB

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-Q6qmGJWF-1669108012106)(C:\Users\vanpersie\AppData\Roaming\Typora\typora-user-images\image-20221119161843466.png)]

点击file->attach to HotSpot process; 然后输入进程id

(3)查找某个对象

打开Tools->FindObjectByQuary

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-x0P87NcB-1669108012106)(C:\Users\vanpersie\AppData\Roaming\Typora\typora-user-images\image-20221119162142767.png)]

查处的是Dog类型对象在内存中的地址

点击这个地址

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-SZ8ndUvG-1669108012106)(C:\Users\vanpersie\AppData\Roaming\Typora\typora-user-images\image-20221119162259041.png)]

点击windows->console;输入下面的命令,然后回车

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-h8Lxl4WH-1669108012107)(C:\Users\vanpersie\AppData\Roaming\Typora\typora-user-images\image-20221119162457225.png)]

点击Tools->inspector, 然后输入地址;即可查看Dog类在java虚拟机中的样子(其中包含了类中所有的信息)

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-RnPXfF43-1669108012107)(C:\Users\vanpersie\AppData\Roaming\Typora\typora-user-images\image-20221119162603974.png)]

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-hwRr6ryS-1669108012107)(C:\Users\vanpersie\AppData\Roaming\Typora\typora-user-images\image-20221119162838053.png)]

根据地址查看vtable(Dog类中所有重写方法的内容)中的内容;vtable和Dog类的地址偏移量是1B8;用1a0+1B8得出的结果是358;

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-oyTBavDt-1669108012108)(C:\Users\vanpersie\AppData\Roaming\Typora\typora-user-images\image-20221119163049996.png)]

输入这个地址; 然后点击tools->Class Browser 输入Dog查看其中方法的地址,发现Command line中的最后一个地址对应eat方法

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-nCvMsCsG-1669108012108)(C:\Users\vanpersie\AppData\Roaming\Typora\typora-user-images\image-20221119163356964.png)]

​ 再来看Animal中的

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-5SKwJMob-1669108012108)(C:\Users\vanpersie\AppData\Roaming\Typora\typora-user-images\image-20221119163656522.png)]

再点击Object查看

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-5MmThjxt-1669108012108)(C:\Users\vanpersie\AppData\Roaming\Typora\typora-user-images\image-20221119163827317.png)]

总结:

因为普通成员方法需要在运行时才能确定具体的内容,所以虚拟机需要调用invokevirtual指令

在执行invokevirtual指令时,经历了以下几个步骤

  • 先通过栈帧中对象的引用找到对象
  • 分析对象头,找到对象实际的Class
  • Class结构中有vtable
  • 查询vtable找到方法的具体地址
  • 执行方法的字节码
2.11. 异常处理
(1). try-catch
public class Demo1 
{
	public static void main(String[] args) 
    {
		int i = 0;
		try 
        {
			i = 10;
		}
        catch (Exception e) 
        {
			i = 20;
		}
	}
}

对应字节码指令

Code:
     stack=1, locals=3, args_size=1
        0: iconst_0
        1: istore_1
        2: bipush        10 //开始执行try中的代码 i=10
        4: istore_1
        5: goto          12 //跳转到12行
        8: astore_2
        9: bipush        20
       11: istore_1
       12: return
     //多出来一个异常表
     Exception table: //会检测从2-4行的代码,不包括5;如果出现异常判断异常种类是否一致,如果一致就会进入到第8行代码
        from    to  target type
            2     5     8   Class java/lang/Exception
  • 可以看到多出来一个 Exception table 的结构,[from, to) 是前闭后开(也就是检测2~4行)的检测范围,一旦这个范围内的字节码执行出现异常,则通过 type 匹配异常类型,如果一致,进入 target 所指示行号
  • 8行的字节码指令 astore_2 是将异常对象引用存入局部变量表的2号位置(为e)
(2). 多个single-catch
public class Demo1 {
	public static void main(String[] args) 
    {
		int i = 0;
		try 
        {
			i = 10;
		}
        catch (ArithmeticException e) 
        {
			i = 20;
		}
        catch (Exception e) 
        {
			i = 30;
		}
	}
}

对应的字节码

Code:
     stack=1, locals=3, args_size=1
        0: iconst_0
        1: istore_1
        2: bipush        10
        4: istore_1
        5: goto          19
        8: astore_2
        9: bipush        20
       11: istore_1
       12: goto          19
       15: astore_2
       16: bipush        30
       18: istore_1
       19: return
     Exception table:
        from    to  target type
            2     5     8   Class java/lang/ArithmeticException
            2     5    15   Class java/lang/Exception
  • 因为异常出现时,只能进入 Exception table 中一个分支,所以局部变量表 slot 2 位置被共用
(3). multi-catch的情况
public class Demo3_11_3
{

    public static void main(String[] args)
    {
        try
        {
            Method test = Demo3_11_3.class.getMethod("test");
            test.invoke(null);
        }
        catch (NoSuchMethodException | IllegalAccessException | InvocationTargetException e) 
        {
            e.printStackTrace();
        }
    }

    public static void test()
    {
        System.out.println("ok");
    }
}
Code:
      stack=3, locals=2, args_size=1
         0: ldc           #2                  // class day3/Demo3_11_3
         2: ldc           #3                  // String test
         4: iconst_0
         5: anewarray     #4                  // class java/lang/Class
         8: invokevirtual #5                  // Method java/lang/Class.getMethod:(Ljava/lang/String;[Ljava/lang/Class;)Ljava/lang/reflect/Method;
        11: astore_1
        12: aload_1
        13: aconst_null
        14: iconst_0
        15: anewarray     #6                  // class java/lang/Object
        18: invokevirtual #7                  // Method java/lang/reflect/Method.invoke:(Ljava/lang/Object;[Ljava/lang/Object;)Ljava/lang/Object;
        21: pop
        22: goto          30
        25: astore_1
        26: aload_1
        27: invokevirtual #11                 // Method java/lang/ReflectiveOperationException.printStackTrace:()V
        30: return
      Exception table:
         from    to  target type
             0    22    25   Class java/lang/NoSuchMethodException
             0    22    25   Class java/lang/IllegalAccessException
             0    22    25   Class java/lang/reflect/InvocationTargetException
(4). finally
public class Demo2 
{
	public static void main(String[] args) 
	{
		int i = 0;
		try 
		{
			i = 10;
		} 
		catch (Exception e) 
		{
			i = 20;
		} 
		finally 
		{
			i = 30;
		}
	}
}

对应字节码

Code:
     stack=1, locals=4, args_size=1
        0: iconst_0
        1: istore_1
        //try块
        2: bipush        10
        4: istore_1
        //try块执行完后,会执行finally    
        5: bipush        30
        7: istore_1
        8: goto          27
       //catch块     
       11: astore_2 //异常信息放入局部变量表的2号槽位
       12: bipush        20
       14: istore_1
       //catch块执行完后,会执行finally        
       15: bipush        30
       17: istore_1
       18: goto          27
       //出现异常,但未被Exception捕获,会抛出其他异常,这时也需要执行finally块中的代码   
       21: astore_3
       22: bipush        30
       24: istore_1
       25: aload_3
       26: athrow  //抛出异常
       27: return
     Exception table:
        from    to  target type
            2     5    11   Class java/lang/Exception
            2     5    21   any
           11    15    21   any

可以看到 finally 中的代码被复制了 3 份,分别放入 try 流程,catch 流程以及 catch剩余的异常类型流程

注意:虽然从字节码指令看来,每个块中都有finally块,但是finally块中的代码只会被执行一次

(5). finally中的return
public class Demo3 
{
	public static void main(String[] args) 
    {
		int i = Demo3.test();
        //结果为20
		System.out.println(i);
	}

	public static int test() 
    {
		int i;
		try 
        {
			i = 10;
			return i;
		} 
        finally 
        {
			i = 20;
			return i;
		}
	}
}

对应字节码

Code:
     stack=1, locals=3, args_size=0
        0: bipush        10
        2: istore_0
        3: iload_0
        4: istore_1  //暂存返回值
        5: bipush        20
        7: istore_0
        8: iload_0
        9: ireturn	//ireturn会返回操作数栈顶的整型值20 (带返回值类型的return)
       //如果出现异常,还是会执行finally块中的内容,没有抛出异常
       10: astore_2
       11: bipush        20
       13: istore_0
       14: iload_0
       15: ireturn	//这里没有athrow了,也就是如果在finally块中如果有返回操作的话,且try块中出现异常,会吞掉异常!
     Exception table:
        from    to  target type
            0     5    10   any
  • 由于 finally 中的 ireturn 被插入了所有可能的流程,因此返回结果肯定以finally的为准
  • 至于字节码中第 2 行,似乎没啥用,且留个伏笔,看下个例子
  • 跟上例中的 finally 相比,发现没有 athrow 了,这告诉我们:如果在 finally 中出现了 return,会吞掉异常
  • 所以不要在finally中进行返回操作
演示被吞掉的异常
public class Demo3 
{
   public static void main(String[] args) 
   {
      int i = Demo3.test();
      //最终结果为20
      System.out.println(i);
   }

   public static int test() 
   {
      int i;
      try 
      {
         i = 10;
         //这里应该会抛出异常
         i = i/0;
         return i;
      } 
      finally 
      {
         i = 20;
         return i;
      }
   }
}

会发现打印结果为20,并未抛出异常

(6). finally不带return
public class Demo4 {
	public static void main(String[] args) 
    {
		int i = Demo4.test();
		System.out.println(i); //10
	}

	public static int test() 
    {
		int i = 10;
		try 
        {
			return i;
		} 
        finally 
        {
			i = 20;
		}
	}
}

对应字节码

Code:
     stack=1, locals=3, args_size=0
        0: bipush        10
        2: istore_0 //赋值给i = 10
        3: iload_0	//加载到操作数栈顶
        4: istore_1 //加载到局部变量表的1号位置; 目的是为了固定返回值;因为try中有return
        5: bipush        20
        7: istore_0 //赋值给i = 20
        8: iload_1 //加载局部变量表1号位置的数10到操作数栈
        9: ireturn //返回操作数栈顶元素 int(10)
       10: astore_2
       11: bipush        20
       13: istore_0
       14: aload_2 //加载异常
       15: athrow //抛出异常
     Exception table:
        from    to  target type
            3     5    10   any
(7). Synchronized
public class Demo5 
{
	public static void main(String[] args) 
    {
		int i = 10;
		Lock lock = new Lock();
		synchronized (lock) 
        {
			System.out.println(i); //10
		}
	}
}

class Lock{}

对应字节码

Code:
     stack=2, locals=5, args_size=1
        0: bipush        10
        2: istore_1
        3: new           #2                  // class com/nyima/JVM/day06/Lock
        6: dup //复制一份,放到操作数栈顶,用于构造函数消耗
        7: invokespecial #3                  // Method com/nyima/JVM/day06/Lock."<init>":()V
       10: astore_2 //剩下的一份放到局部变量表的2号位置
       11: aload_2 //加载到操作数栈
       12: dup //复制一份,放到操作数栈,用于加锁时消耗
       13: astore_3 //将操作数栈顶元素弹出,暂存到局部变量表的三号槽位。这时操作数栈中有一份对象的引用
       14: monitorenter //加锁 (lock引用)
       //锁住后代码块中的操作    
       15: getstatic     #4                  // Field java/lang/System.out:Ljava/io/PrintStream;
       18: iload_1
       19: invokevirtual #5                  // Method java/io/PrintStream.println:(I)V
       //如果没有异常;加载局部变量表中三号槽位对象的引用,用于解锁    
       22: aload_3    
       23: monitorexit //解锁 (lock引用)
       24: goto          34
       //出现异常操作    
       27: astore        4
       29: aload_3
       30: monitorexit //解锁 (lock引用)
       31: aload         4
       33: athrow	//抛出异常
       34: return
     //可以看出,无论何时出现异常,都会跳转到27行,将异常放入局部变量中,并进行解锁操作,然后加载异常并抛出异常。      
     Exception table:
        from    to  target type
           15    24    27   any
           27    31    27   any

注意:

方法级别的synchronized 不会在字节码指令中有所体现

3、编译期处理

所谓的 语法糖 ,其实就是指 java 编译器把 *.java 源码编译为 *.class 字节码的过程中,自动生成转换的一些代码,主要是为了减轻程序员的负担,算是 java 编译器给我们的一个额外福利

注意,以下代码的分析,借助了 javap 工具,idea 的反编译功能,idea 插件 jclasslib 等工具。另外, 编译器转换的结果直接就是 class 字节码,只是为了便于阅读,给出了 几乎等价 的 java 源码方式,并不是编译器还会转换出中间的 java 源码,切记。

3.1. 默认构造函数
public class Candy1 {}

经过编译期优化后

public class Candy1 
{
   //这个无参构造器是java编译器帮我们加上的
   public Candy1() 
   {
      //即调用父类 Object 的无参构造方法,即调用 java/lang/Object." <init>":()V
      super();
   }
}
3.2. 自动拆装箱
  • 基本类型赋值给包装类型,称为装箱

  • 包装类型赋值给基本类型,称谓拆箱

在JDK 5以后,它们的转换可以在编译期自动完成

public class Demo2 
{
   public static void main(String[] args) 
   {
      Integer x = 1;
      int y = x;
   }
}

转换过程如下; 这段代码在 JDK 5 之前是无法编译通过的,必须改写下面这样;

public class Demo2 
{
   public static void main(String[] args) 
   {
      //基本类型赋值给包装类型,称为装箱
      Integer x = Integer.valueOf(1);
      //包装类型赋值给基本类型,称谓拆箱
      int y = x.intValue();
   }
}
3.3. 泛型集合取值

泛型也是在 JDK 5 开始加入的特性,但 java 在编译泛型代码后会执行 泛型擦除 的动作,即泛型信息在编译为字节码之后就丢失了,实际的类型都当做了 Object 类型来处理:

public class Candy3 
{
    public static void main(String[] args) 
    {
        List<Integer> list = new ArrayList<>();
        list.add(10); // 实际调用的是 List.add(Object e)
        Integer x = list.get(0); // 实际调用的是 Object obj = List.get(int index);
    }
}

对应字节码

Code:
    stack=2, locals=3, args_size=1
       0: new           #2                  // class java/util/ArrayList
       3: dup
       4: invokespecial #3                  // Method java/util/ArrayList."<init>":()V
       7: astore_1
       8: aload_1
       9: bipush        10
      11: invokestatic  #4                  // Method java/lang/Integer.valueOf:(I)Ljava/lang/Integer;
      //这里进行了泛型擦除,实际调用的是add(Objcet o); 调用的全是Object类型
      14: invokeinterface #5,  2            // InterfaceMethod java/util/List.add:(Ljava/lang/Object;)Z

      19: pop
      20: aload_1
      21: iconst_0
      //这里也进行了泛型擦除,实际调用的是get(Object o)   
      22: invokeinterface #6,  2            // InterfaceMethod java/util/List.get:(I)Ljava/lang/Object;
	  //这里进行了类型转换,将Object转换成了Integer
      27: checkcast     #7                  // class java/lang/Integer
      30: astore_2
      31: return

所以调用get函数取值时,有一个类型转换的操作

Integer x = (Integer) list.get(0);

如果要将返回结果赋值给一个int类型的变量,则还有自动拆箱的操作

int x = (Integer) list.get(0).intValue();
3.4. 可变参数
public class Demo4 
{
   public static void foo(String... args) 
   {
      //将args赋值给arr,可以看出String...实际就是String[] 
      String[] arr = args; //直接赋值
      System.out.println(arr.length);
   }

   public static void main(String[] args) 
   {
      foo("hello", "world");
   }
}

可变参数 String… args 其实是一个 String[] args ,从代码中的赋值语句中就可以看出来。 同 样 java 编译器会在编译期间将上述代码变换为:

public class Demo4 
{
   public Demo4 {}

   public static void foo(String[] args) 
   {
      String[] arr = args;
      System.out.println(arr.length);
   }

   public static void main(String[] args) 
   {
      foo(new String[]{"hello", "world"});
   }
}

注意,如果调用的是foo(),即未传递参数时,等价代码为foo(new String[]{}),创建了一个空数组,而不是直接传递的null

3.5. foreach

仍是 JDK 5 开始引入的语法糖,数组的循环:

public class Demo5 
{
	public static void main(String[] args) 
	{
        //数组赋初值的简化写法也是一种语法糖。
		int[] arr = {1, 2, 3, 4, 5};
		for(int x : arr) 
		{
			System.out.println(x);
		}
	}
}

编译器会帮我们转换为

public class Demo5 
{
    public Demo5 {}

	public static void main(String[] args) 
	{
		int[] arr = new int[]{1, 2, 3, 4, 5};
		for(int i=0; i<arr.length; ++i) 
		{
			int x = arr[i];
			System.out.println(x);
		}
	}
}

如果是集合使用foreach

public class Demo5 
{
   public static void main(String[] args) 
   {
      List<Integer> list = Arrays.asList(1, 2, 3, 4, 5);
      for (Integer x : list) 
      {
         System.out.println(x);
      }
   }
}

实际被编译器转换为对迭代器的调用:

public class Demo5 
{
   public Demo5 {}
    
   public static void main(String[] args) 
   {
        List<Integer> list = Arrays.asList(1, 2, 3, 4, 5);
        Iterator iter = list.iterator();
        while(iter.hasNext()) 
        {
            Integer e = (Integer)iter.next();
            System.out.println(e);
        }
    }
}

注意 :foreach 循环写法,能够配合数组,以及所有实现了 Iterable 接口的集合类一起使用,其 中 Iterable 用来获取集合的迭代器( Iterator

3.6. switch字符串

从 JDK 7 开始,switch 可以作用于字符串和枚举类,这个功能其实也是语法糖,例如:

public class Demo6 
{
   public static void main(String[] args) 
   {
      String str = "hello";
      switch (str) 
      {
         case "hello" :
            System.out.println("h");
            break;
         case "world" :
            System.out.println("w");
            break;
         default:
            break;
      }
   }
}

在编译器中执行的操作

public class Demo6 
{
   public Demo6(){}
   
   public static void main(String[] args) 
   {
      String str = "hello";
      int x = -1;
      //通过字符串的hashCode+value来判断是否匹配
      switch (str.hashCode()) 
      {
         //hello的hashCode
         case 99162322 :
            //再次比较,因为字符串的hashCode有可能相等
            if(str.equals("hello")) 
            {
               x = 0;
            }
            break;
         //world的hashCode
         case 11331880 :
            if(str.equals("world")) 
            {
               x = 1;
            }
            break;
         default:
            break;
      }

      //用第二个switch在进行输出判断
      switch (x) 
      {
         case 0:
            System.out.println("h");
            break;
         case 1:
            System.out.println("w");
            break;
         default:
            break;
      }
   }
}

过程说明:

  • 在编译期间,单个的switch被分为了两个
    • 第一个用来匹配字符串,并给x赋值

      • 字符串的匹配用到了字符串的hashCode,还用到了equals方法
      • 使用hashCode是为了提高比较效率,使用equals是防止有hashCode冲突(如BM和C. 的hashCode值相同)
    • 第二个用来根据x的值来决定输出语句

hashCode值相同的情况:

public class Candy6_1 
{
    public static void choose(String str) 
    {
        switch (str) 
        {
            case "BM": 
            {
                System.out.println("h");
                break;
            }
            case "C.": 
            {
                System.out.println("w");
                break;
            }
        }
    }
}

会被编译器转换为:

public class Candy6_1 
{
    public Candy6_1(){}

    public static void choose(String var0) 
    {
        byte var2 = -1;
        switch(var0.hashCode()) 
        {
        case 2123:
            if (var0.equals("C.")) 
            {
                var2 = 1;
            } else if (var0.equals("BM")) 
            {
                var2 = 0;
            }
        default:
            switch(var2) 
            {
                case 0:
                    System.out.println("h");
                    break;
                case 1:
                    System.out.println("w");
            }
        }
    }
}
3.7. switch枚举
public enum SEX 
{
   MALE, FEMALE;
}

public class Demo7 
{
   public static void main(String[] args) 
   {
      SEX sex = SEX.MALE;
      switch (sex) 
      {
         case MALE:
            System.out.println("man");
            break;
         case FEMALE:
            System.out.println("woman");
            break;
         default:
            break;
      }
   }
}

编译器中执行的代码如下

public class Demo7 
{
   /**     
    * 定义一个合成类(仅 jvm 使用,对我们不可见)     
    * 用来映射枚举的 ordinal 与数组元素的关系     
    * 枚举的 ordinal 表示枚举对象的序号,从 0 开始     
    * 即 MALE 的 ordinal()=0,FEMALE 的 ordinal()=1     
    */ 
   static class $MAP 
   {
      //数组大小即为枚举元素个数,里面存放了case用于比较的数字
      static int[] map = new int[2];
      static 
      {
         //ordinal即枚举元素对应所在的位置,MALE为0,FEMALE为1
         map[SEX.MALE.ordinal()] = 1;
         map[SEX.FEMALE.ordinal()] = 2;
      }
   }

   public static void main(String[] args) 
   {
      SEX sex = SEX.MALE;
      //将对应位置枚举元素的值赋给x,用于case操作
      int x = $MAP.map[sex.ordinal()];
      switch (x) 
      {
         case 1:
            System.out.println("man");
            break;
         case 2:
            System.out.println("woman");
            break;
         default:
            break;
      }
   }
}
3.8. 枚举类

JDK 7 新增了枚举类,以前面的性别枚举为例:

enum SEX 
{
   MALE, FEMALE;
}

转换后的代码

public final class Sex extends Enum<Sex> 
{   
   //对应枚举类中的元素
   public static final Sex MALE;    
   public static final Sex FEMALE;    
   private static final Sex[] $VALUES;
   
   static 
   {       
       //调用构造函数,传入枚举元素的值及ordinal
       MALE = new Sex("MALE", 0);    
       FEMALE = new Sex("FEMALE", 1);   
       $VALUES = new Sex[]{MALE, FEMALE}; 
   }
 	
    //调用父类中的方法
    private Sex(String name, int ordinal) 
    {     
        super(name, ordinal);    
    }

    public static Sex[] values() 
    {  
        return $VALUES.clone();  
    }

    public static Sex valueOf(String name) 
    { 
        return Enum.valueOf(Sex.class, name);  
    } 
}
3.9. try-with-resources

JDK 7开始新增了对需要关闭的资源处理的特殊语法try-with-resources:

try(资源变量 = 创建资源对象)
{}
catch()
{}

其中资源对象需要实现AutoCloseable接口,例如InputStream、OutputStream、Connection、Statement、ResultSet 等接口实现了AutoCloseable,使用try-with-resource 可以不用写finally 语句块,编译器会帮助生成关闭资源代码,例如:

public class Candy9
{
    public static void main(String[] args)
    {
        try(InputStream is = new FileInputStream("d:\\1.txt"))
        {
            System.out.println(is);
        }
        catch (IOException e)
        {
            e.printStackTrace();
        }
    }
}

会被转化为:

public class Candy9
{
    public Candy9(){}

    public static void main(String[] args)
    {
        try
        {
            InputStream is = new FileInputStream("d:\\1.txt");
            Throwable t = null;
            try
            {
                System.out.println(is);
            }
            catch (Throwable e1)
            {
                //t 是我们代码出现的异常
                t = e1;
                throw e1;
            }
            finally
            {
                //判断了资源不为空
                if(is != null)
                {
                    //如果我们代码有异常
                    if(t != null)
                    {
                        try
                        {
                            is.close();
                        }
                        catch (Throwable e2)
                        {
                            //如果close过程中出现异常,做为被压制异常添加
                            t.addSuppressed(e2);
                        }
                    }
                    else
                    {
                        //如果我们代码没有异常,close过程中如果出现异常就是最后catch 块中的e
                        is.close();
                    }
                }
            }
        }
        catch (IOException e)
        {
            e.printStackTrace();
        }
    }
}

为什么要设计一个addSuppressed (Throwable e)(添加被压制异常)的方法呢?是为了防止异常信息的丢失。

3.10. 方法重写时的桥接方法

我们都知道,方法重写时返回值分为两种情况:

  • 父子类的返回值完全一致
  • 子类返回值可以是父类返回值的子类(看下面的例子)
class A
{
	public Number m()
	{
		return 1;
	}
}

class B extends A
{
    @Override
    //子类 m 的方法的返回值是 Integer 是父类 m方法返回值 Number的子类
    public Integer m()
    {
        return 2;
    }
}

对于子类,java编译器会做以下处理

class B extends A
{
    public Integer m()
    {
        return 2;
    }
    
    //此方法才是真正重写了父类 public Number m() 方法
    public synthetic bridge Number m()
    {
        //调用public Integer m()
        return m();
    }
}

其中桥接方法比较特殊,仅对java虚拟机可见。

3.11. 匿名内部类
public class Demo8 
{
   public static void main(String[] args) 
   {
      Runnable runnable = new Runnable() 
      {
         @Override
         public void run() 
         {
            System.out.println("running...");
         }
      };
   }
}

转换后的代码

public class Demo8 
{
   public static void main(String[] args) 
   {
      //用额外创建的类来创建匿名内部类对象
      Runnable runnable = new Demo8$1();
   }
}

//创建了一个额外的类,实现了Runnable接口
final class Demo8$1 implements Runnable 
{
   public Demo8$1() {}

   public void run() 
   {
      System.out.println("running...");
   }
}

如果匿名内部类中引用了局部变量

public class Demo8 
{
   public static void test(final int x) 
   {
      int x = 1;
      Runnable runnable = new Runnable() 
      {
         @Override
         public void run() 
         {
            System.out.println(x);
         }
      };
   }
}

转化后代码

public class Demo8 
{
   public static void test(final int x)  
   {
      //用额外创建的类来创建匿名内部类对象
      Runnable runnable = new Demo8$1(x);
   }
}

final class Demo8$1 implements Runnable 
{
   //多创建了一个变量
   int val$x;
    
   //变为了有参构造器
   Demo8$1(int x) 
   {
      this.val$x = x;
   }

   public void run() 
   {
      System.out.println(val$x);
   }
}

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-hFFoJ9q6-1669108012110)(C:\Users\vanpersie\AppData\Roaming\Typora\typora-user-images\image-20221121114114762.png)]

4、类加载阶段

4.1. 加载
  • 将类的字节码载入方法区(1.8后为元空间,在本地内存中)中,内部采用 C++ 的 instanceKlass 描述 java 类,它的重要 field 有:

    • _java_mirror 即 java 的类镜像,例如对 String 来说,它的镜像类就是 String.class,作用是把 klass 暴露给 java 使用
    • _super 即父类
    • _fields 即成员变量
    • _methods 即方法
    • _constants 即常量池
    • _class_loader 即类加载器
    • _vtable 虚方法表
    • _itable 接口方法
  • 如果这个类还有父类没有加载,先加载父类

  • 加载和链接可能是交替运行

  • instanceKlass保存在方法区。JDK 8以后,方法区位于元空间中,而元空间又位于本地内存中

  • _java_mirror则是保存在堆内存

  • InstanceKlass和*.class(JAVA镜像类)互相保存了对方的地址

  • 类的对象在对象头中保存了*.class的地址。让对象可以通过其找到方法区中的instanceKlass,从而获取类的各种信息

img

4.2. 链接

链接分为三个小的步骤:验证、准备和解析;

验证

验证类是否符合 JVM规范,安全性检查

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-jPnBqPWR-1669108012110)(C:\Users\vanpersie\AppData\Roaming\Typora\typora-user-images\image-20221121121412997.png)]

准备

为 static 变量分配空间,设置默认值

  • static变量在JDK 7以前是存储与instanceKlass末尾。但在JDK 7以后就存储在_java_mirror末尾了
  • static变量在分配空间和赋值是在两个阶段完成的。分配空间在准备阶段完成,赋值在初始化阶段完成 (如图中a和b)
  • 如果 static 变量是 final 的基本类型,以及字符串常量,那么编译阶段值就已经确定了,赋值在准备阶段完成 (如图中c和d)
  • 如果 static 变量是 final 的,但属于引用类型,那么赋值也会在初始化阶段完成 (如图中e)

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-mQKNYSev-1669108012111)(C:\Users\vanpersie\AppData\Roaming\Typora\typora-user-images\image-20221121122144607.png)]

解析

解析的含义

将常量池中的符号引用解析为直接引用

  • 未解析时,常量池中的看到的对象仅是符号,未真正的存在于内存中
public class Demo1 
{
   public static void main(String[] args) throws IOException, ClassNotFoundException 
   {
      ClassLoader loader = Demo1.class.getClassLoader();
      //只加载 不解析和初始化 C
      Class<?> c = loader.loadClass("com.nyima.JVM.day8.C");
       
       //加载、解析、初始化C
       //new C();
      //用于阻塞主线程
      System.in.read();
   }
}

class C 
{
   D d = new D();
}

class D 
{

}
  • 打开HSDB

    • 先获得要查看的进程ID
    jps
    
    • 在JDK目录下打开HSDB

      [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-rjkQSKJk-1669108012111)(C:\Users\vanpersie\AppData\Roaming\Typora\typora-user-images\image-20221121122700580.png)]

    • 定位需要的进程

    img

    img](

    • 可以看到此时只加载了类C

img

​ 查看类C的常量池,可以看到类D未被解析,只是存在于常量池中的符号

img

  • 解析以后,会将常量池中的符号引用解析为直接引用

    • 可以看到,此时已加载并解析了类C和类D

    img

img

4.3. 初始化

初始化阶段就是执行类构造器cinit()方法的过程,虚拟机会保证这个类的『构造方法』的线程安全

  • cinit()方法是由编译器自动收集类中的所有类变量的赋值动作和静态语句块(static{}块)中的语句合并产生的

注意

编译器收集的顺序是由语句在源文件中出现的顺序决定的,静态语句块中只能访问到定义在静态语句块之前的变量,定义在它之后的变量,在前面的静态语句块可以赋值,但是不能访问,如

img

发生时机

类的初始化的懒惰的,以下情况会初始化

  • main 方法所在的类,总会被首先初始化
  • 首次访问这个类的静态变量或静态方法时
  • 子类初始化,如果父类还没初始化,会引发
  • 子类访问父类的静态变量,只会触发父类的初始化
  • Class.forName 会初始化
  • new 会导致初始化

以下情况不会初始化

  • 访问类的 static final 静态常量(基本类型和字符串)

  • 类对象.class 不会触发初始化

  • 创建该类对象的数组

  • 类加载器的.loadClass方法

  • Class.forName 的参数2为false时

验证类是否被初始化,可以看该类的静态代码块是否被执行

这里用一个例子来验证 “发生时机” 中的观点:(实验时请先全部注释,每次只执行其中一个)

public class Load3 
{
    static 
    {
        System.out.println("main init");
    }
    
    public static void main(String[] args) throws ClassNotFoundException 
    {
        // 1. static final 静态常量(基本类型和字符串)不会触发初始化
        System.out.println(B.b);
        // 2. 类对象.class 不会触发初始化
        System.out.println(B.class);
        // 3. 创建该类的数组不会触发初始化
        System.out.println(new B[0]);
        // 4. 不会初始化类 B,但会加载 B、A
        ClassLoader cl = Thread.currentThread().getContextClassLoader();
        cl.loadClass("cn.itcast.jvm.t3.B");
        // 5. 不会初始化类 B,但会加载 B、A
        ClassLoader c2 = Thread.currentThread().getContextClassLoader();
        Class.forName("cn.itcast.jvm.t3.B", false, c2);
        
        // 1. 首次访问这个类的静态变量或静态方法时
        System.out.println(A.a);
        // 2. 子类初始化,如果父类还没初始化,会引发
        System.out.println(B.c);
        // 3. 子类访问父类静态变量,只触发父类初始化
        System.out.println(B.a);
        // 4. 会初始化类 B,并先初始化类 A
        Class.forName("cn.itcast.jvm.t3.B");
    }
}

class A 
{
    static int a = 0;
    static 
    {
        System.out.println("a init");
    }
}

class B extends A 
{
    final static double b = 5.0;
    static boolean c = false;
    static 
    {
        System.out.println("b init");
    }
}
4.4. 练习
练习1

从字节码分析,使用a、b、c这三个常量是否会导致E初始化

public class Load4 
{
    public static void main(String[] args) 
    {
        System.out.println(E.a); //不会
        System.out.println(E.b); //不会
        System.out.println(E.c); //会(是包装类型,所以会触发初始化)
    }
}

class E 
{
    public static final int a = 10;
    public static final String b = "hello";
    public static final Integer c = 20;  // Integer.valueOf(20)
    static 
    {
        System.out.println("init E");
    }
}

对应的字节码

D:\Ajvm\out\production\jvm\day3>javap -v E.class
...
{
  public static final int a; //a的值在准备阶段已经确定
    descriptor: I
    flags: ACC_PUBLIC, ACC_STATIC, ACC_FINAL
    ConstantValue: int 10

  public static final java.lang.String b; //b的值在准备阶段已经确定
    descriptor: Ljava/lang/String;
    flags: ACC_PUBLIC, ACC_STATIC, ACC_FINAL
    ConstantValue: String hello

  public static final java.lang.Integer c; //c的值在初始化阶段才确定
    descriptor: Ljava/lang/Integer;
    flags: ACC_PUBLIC, ACC_STATIC, ACC_FINAL

  day3.E();
    descriptor: ()V
    flags:
    Code:
      stack=1, locals=1, args_size=1
         0: aload_0
         1: invokespecial #1                  // Method java/lang/Object."<init>":()V
         4: return
      LineNumberTable:
        line 13: 0
      LocalVariableTable:
        Start  Length  Slot  Name   Signature
            0       5     0  this   Lday3/E;

  static {};
    descriptor: ()V
    flags: ACC_STATIC
    Code:
      stack=2, locals=0, args_size=0	//c的值被确定
         0: bipush        20
         2: invokestatic  #2                  // Method java/lang/Integer.valueOf:(I)Ljava/lang/Integer;
         5: putstatic     #3                  // Field c:Ljava/lang/Integer;
         8: getstatic     #4                  // Field java/lang/System.out:Ljava/io/PrintStream;
        11: ldc           #5                  // String init E
        13: invokevirtual #6                  // Method java/io/PrintStream.println:(Ljava/lang/String;)V
        16: return
      LineNumberTable:
        line 17: 0
        line 20: 8
        line 21: 16
}
SourceFile: "Load4.java"
练习2

典型引用—完成懒惰初始化单例模式

public class Load9
{
    public static void main(String[] args)
    {
//        Singleton.test(); //不会触发lazy holder init; 因为没有用到getInstance
        Singleton.getInstance(); //会触发lazy holder init
    }

}

class Singleton
{
    public static void test()
    {
        System.out.println("test");
    }

    private Singleton() {}

    private static class LazyHolder
    {
        private static final Singleton SINGLETON = new Singleton();
        static
        {
            System.out.println("lazy holder init");
        }
    }

    public static Singleton getInstance()
    {
        return LazyHolder.SINGLETON;
    }
}

以上的实现特点是:

  • 懒惰实例化
  • 初始化的线程安全有保障

5、类加载器

Java虚拟机设计团队有意把类加载阶段中的**“通过一个类的全限定名来获取描述该类的二进制字节流”这个动作放到Java虚拟机外部去实现,以便让应用程序自己决定如何去获取所需的类。实现这个动作的代码被称为“类加载器”**(ClassLoader)

类与类加载器

类加载器虽然只用于实现类的加载动作,但它在Java程序中起到的作用却远超类加载阶段

对于任意一个类,都必须由加载它的类加载器和这个类本身一起共同确立其在Java虚拟机中的唯一性,每一个类加载器,都拥有一个独立的类名称空间。这句话可以表达得更通俗一些:比较两个类是否“相等”,只有在这两个类是由同一个类加载器加载的前提下才有意义,否则,即使这两个类来源于同一个Class文件,被同一个Java虚拟机加载,只要加载它们的类加载器不同,那这两个类就必定不相等

以JDK 8为例

名称加载的类说明
Bootstrap ClassLoader(启动类加载器)JAVA_HOME/jre/lib无法直接访问
Extension ClassLoader (拓展类加载器)JAVA_HOME/jre/lib/ext上级为Bootstrap,显示为null
Application ClassLoader (应用程序类加载器)classpath上级为Extension
自定义类加载器自定义上级为Application
5.1. 启动类加载器

可通过在控制台输入指令,使得类被启动类加器加载

Bootstrap 类加载器加载 F 类:

package day3.load.F;
public class F 
{
    static 
    {
        System.out.println("bootstrap F init");
    }
}

执行

package day3.load.F;
public class Load5_1 
{
    public static void main(String[] args) throws ClassNotFoundException 
    {
        Class<?> aClass = Class.forName("day3.load.F");
        System.out.println(aClass.getClassLoader());
    }
}

通过终端命令让代码输出

D:\Ajvm\out\production\jvm>java -Xbootclasspath/a:. day3.load.Load5_1 //命令
bootstrap F init //执行结果,先初始化
null //由于打印的结果是null,所以可以判断F这个类是由启动类加载器加载的。(因为启动类加载器中是C++编写的,java代码无法访问,所以得到的是null)
  • -Xbootclasspath 表示设置 bootclasspath
  • 其中 /a:. 表示将当前目录追加至 bootclasspath 之后
  • 可以用这个办法替换核心类
    • java -Xbootclasspath:
    • 也可以追加
      • java -Xbootclasspath/a:<追加路径>(后追加)
      • java -Xbootclasspath/p:<追加路径>(前追加)
5.2. 拓展类加载器

如果classpath和JAVA_HOME/jre/lib/ext 下有同名类,加载时会使用拓展类加载器加载。当应用程序类加载器发现拓展类加载器已将该同名类加载过了,则不会再次加载

Extension 类加载器加载 G类:

public class G
{
    static
    {
        System.out.println("G init");
    }
}

执行

  • 此时运行,会是应用程序类加载器加载的;但是我们将 G 这个java代码打包

    [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-7DZ0rVnw-1669108012112)(C:\Users\vanpersie\AppData\Roaming\Typora\typora-user-images\image-20221121143853133.png)]

  • 将打包好的my.jar放到jdk文件夹下的jre/lib/ext文件夹中

    [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-P1QiKp3z-1669108012113)(C:\Users\vanpersie\AppData\Roaming\Typora\typora-user-images\image-20221121143928573.png)]

  • 此时我们再去运行得到的会是拓展类加载器加载的。因为jvm在执行时,会先判断是否被拓展类加载器执行,如果没有被执行,才会去执行应用程序类加载器;如果有则不会执行。

/**
 * 演示 拓展类加载器
 * 里面也有一个 G 的类,观察到底是哪个类被加载了; 
 */
public class Load5_2
{
    public static void main(String[] args) throws ClassNotFoundException
    {
        Class<?> aClass = Class.forName("day3.load.G");
        System.out.println(aClass.getClassLoader());
    }
}

输出

G init
sun.misc.Launcher$ExtClassLoader@18b4aac2 //由拓展类加载器执行
5.3. 双亲委派模式
  • 当AppClassLoader加载一个class时,它首先不会自己去尝试加载这个类,而是把类加载请求委派给父类加载器ExtClassLoader去完成。

  • 当ExtClassLoader加载一个class时,它首先也不会自己去尝试加载这个类,而是把类加载请求委派给BootStrapClassLoader去完成。

  • 如果BootStrapClassLoader加载失败(例如在$JAVA_HOME/jre/lib里未查找到该class),会使用ExtClassLoader来尝试加载;

  • 若ExtClassLoader也加载失败,则会使用AppClassLoader来加载,如果AppClassLoader也加载失败,则会报出异常ClassNotFoundException。

所谓的双亲委派,就是指调用类加载器的 loadClass 方法时,查找类的规则

loadClass源码

protected Class<?> loadClass(String name, boolean resolve) throws ClassNotFoundException 
{
    synchronized (getClassLoadingLock(name)) 
    {
        // 1. 检查该类是否已经加载
        Class<?> c = findLoadedClass(name);
        if (c == null) 
        {
            long t0 = System.nanoTime();
            try 
            {
                if (parent != null) 
                {
                    // 2. 有上级的话,委派上级 loadClass
                    c = parent.loadClass(name, false);
                } 
                else 
                {
                    // 3. 如果没有上级了(说明已经到ExtClassLoader),则委派BootstrapClassLoader
                    c = findBootstrapClassOrNull(name);
                }
            } 
            catch (ClassNotFoundException e) {
                
            }
            
            if (c == null) 
            {
                long t1 = System.nanoTime();
                // 4. 每一层找不到,调用 findClass 方法(每个类加载器自己扩展)来加载
                c = findClass(name);
                // 5. 记录耗时
                sun.misc.PerfCounter.getParentDelegationTime().addTime(t1 - t0);
                sun.misc.PerfCounter.getFindClassTime().addElapsedTimeFrom(t1);
                sun.misc.PerfCounter.getFindClasses().increment();
            }
        }
        if (resolve) 
        {
            resolveClass(c);
        }
        return c;
    }
}
5.4. 应用上下文类加载器(还需要重点看)

我们在使用JDBC时,都需要加载Driver驱动,不知道你注意没有,不写

Class.forName("com.mysql.jdbc.Driver")

也可以让com.mysql.jdbc.Driver正确加载,你知道是怎么做的吗?

让我们追踪一下源码:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-Y6C30mOj-1669108012113)(C:\Users\vanpersie\AppData\Roaming\Typora\typora-user-images\image-20221121150638432.png)]

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-HZxClno7-1669108012113)(C:\Users\vanpersie\AppData\Roaming\Typora\typora-user-images\image-20221121150704740.png)]

private static void loadInitialDrivers() {
    String drivers;
    try {
        drivers = AccessController.doPrivileged(new PrivilegedAction<String>() {
            public String run() {
                return System.getProperty("jdbc.drivers");
            }
        });
    } catch (Exception ex) {
        drivers = null;
    }
  
    //(1)使用 ServiceLoader 机制加载驱动,即SPI
    AccessController.doPrivileged(new PrivilegedAction<Void>() {
        public Void run() {

            ServiceLoader<Driver> loadedDrivers = ServiceLoader.load(Driver.class);
            Iterator<Driver> driversIterator = loadedDrivers.iterator();

            try{
                while(driversIterator.hasNext()) {
                    driversIterator.next();
                }
            } catch(Throwable t) {
            // Do nothing
            }
            return null;
        }
    });

    println("DriverManager.initialize: jdbc.drivers = " + drivers);

    //(2)使用jdbc.drivers定义的驱动加载驱动
    if (drivers == null || drivers.equals("")) {
        return;
    }
    String[] driversList = drivers.split(":");
    println("number of Drivers:" + driversList.length);
    for (String aDriver : driversList) {
        try {
            println("DriverManager.Initialize: loading " + aDriver);
            //这里的 ClassLoader.getSystemClassLoader()就是应用程序类加载器
            Class.forName(aDriver, true, ClassLoader.getSystemClassLoader());
        } catch (Exception ex) {
            println("DriverManager.Initialize: load failed: " + ex);
        }
    }
}

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-wfUwXRd4-1669108012113)(C:\Users\vanpersie\AppData\Roaming\Typora\typora-user-images\image-20221121151541281.png)]

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-IKt1y5Yr-1669108012113)(C:\Users\vanpersie\AppData\Roaming\Typora\typora-user-images\image-20221121151606411.png)]

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-MlgjqSEW-1669108012114)(C:\Users\vanpersie\AppData\Roaming\Typora\typora-user-images\image-20221121151449979.png)]

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-cdd2wCpj-1669108012114)(C:\Users\vanpersie\AppData\Roaming\Typora\typora-user-images\image-20221121151623528.png)]

private S nextService() {
    if (!hasNextService())
        throw new NoSuchElementException();
    String cn = nextName;
    nextName = null;
    Class<?> c = null;
    try {
        c = Class.forName(cn, false, loader);
    } catch (ClassNotFoundException x) {
        fail(service,
             "Provider " + cn + " not found");
    }
    if (!service.isAssignableFrom(c)) {
        fail(service,
             "Provider " + cn  + " not a subtype");
    }
    try {
        S p = service.cast(c.newInstance());
        providers.put(cn, p);
        return p;
    } catch (Throwable x) {
        fail(service,
             "Provider " + cn + " could not be instantiated",
             x);
    }java
    throw new Error();          // This cannot happen
}
5.5. 自定义类加载器(还需要重点看)

使用场景

  • 想加载非 classpath 随意路径中的类文件

  • 通过接口来使用实现,希望解耦时,常用在框架设计

  • 这些类希望予以隔离,不同应用的同名类都可以加载,不冲突,常见于 tomcat 容器

步骤

  • 继承ClassLoader父类
  • 要遵从双亲委派机制,重写 findClass 方法
    • 不是重写loadClass方法,否则不会走双亲委派机制
  • 读取类文件的字节码
  • 调用父类的 defineClass 方法来加载类
  • 使用者调用该类加载器的 loadClass 方法

代码

MapImpl1 和 MapImpl2 是创建的两个代码相同的文件

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-OzIyLwdB-1669108012114)(C:\Users\vanpersie\AppData\Roaming\Typora\typora-user-images\image-20221121154326972.png)]

public class Load7
{
    public static void main(String[] args) throws Exception
    {
        MyClassLoader classLoader = new MyClassLoader();
        Class<?> c1 = classLoader.loadClass("MapImpl1");
        Class<?> c2 = classLoader.loadClass("MapImpl1");
        System.out.println(c1 == c2); //true

        MyClassLoader classLoader2 = new MyClassLoader();
        Class<?> c3 = classLoader2.loadClass("MapImpl1");
        System.out.println(c1 == c3); //false 不是同一个类加载器对象

        c1.newInstance();
    }
}

class MyClassLoader extends ClassLoader
{
    @Override // name 就是类名称
    protected Class<?> findClass(String name) throws ClassNotFoundException
    {
        String path = "e:\\myclasspath\\" + name + ".class";

        try
        {
            ByteArrayOutputStream os = new ByteArrayOutputStream();
            Files.copy(Paths.get(path), os);

            // 得到字节数组
            byte[] bytes = os.toByteArray();

            // byte[] -> *.class
            return defineClass(name, bytes, 0, bytes.length);
        } 
        catch (IOException e) 
        {
            e.printStackTrace();
            throw new ClassNotFoundException("类文件未找到", e);
        }
    }
}
破坏双亲委派模式
  • 双亲委派模型的第一次“被破坏”其实发生在双亲委派模型出现之前——即JDK1.2面世以前的“远古”时代
    • 建议用户重写findClass()方法,在类加载器中的loadClass()方法中也会调用该方法
  • 双亲委派模型的第二次“被破坏”是由这个模型自身的缺陷导致的
    • 如果有基础类型又要调用回用户的代码,此时也会破坏双亲委派模式
  • 双亲委派模型的第三次“被破坏”是由于用户对程序动态性的追求而导致的
    • 这里所说的“动态性”指的是一些非常“热”门的名词:代码热替换(Hot Swap)、模块热部署(Hot Deployment)等

6、运行期优化

6.1. 分层编译

JVM 将执行状态分成了 5 个层次:

  • 0层:解释执行,用解释器将字节码翻译为机器码
  • 1层:使用 C1 即时编译器编译执行(不带 profiling)
  • 2层:使用 C1 即时编译器编译执行(带基本的profiling)
  • 3层:使用 C1 即时编译器编译执行(带完全的profiling)
  • 4层:使用 C2 即时编译器编译执行

profiling 是指在运行过程中收集一些程序执行状态的数据,例如【方法的调用次数】,【循环的 回边次数】等

(1). 即时编译器(JIT)与解释器的区别
  • 解释器
    • 将字节码解释为机器码,下次即使遇到相同的字节码,仍会执行重复的解释
    • 是将字节码解释为针对所有平台都通用的机器码
  • 即时编译器
    • 将一些字节码编译为机器码,并存入 Code Cache,下次遇到相同的代码,直接执行,无需再编译
    • 根据平台类型,生成平台特定的机器码

对于大部分的不常用的代码,我们无需耗费时间将其编译成机器码,而是采取解释执行的方式运行;另一方面,对于仅占据小部分的热点代码,我们则可以将其编译成机器码,以达到理想的运行速度。 执行效率上简单比较一下 Interpreter < C1 < C2,总的目标是发现热点代码(hotspot名称的由 来),并优化这些热点代码

(2). 逃逸分析

逃逸分析(Escape Analysis)简单来讲就是,Java Hotspot 虚拟机可以分析新创建对象的使用范围,并决定是否在 Java 堆上分配内存的一项技术

逃逸分析的 JVM 参数如下:

  • 开启逃逸分析:-XX:+DoEscapeAnalysis
  • 关闭逃逸分析:-XX:-DoEscapeAnalysis
  • 显示分析结果:-XX:+PrintEscapeAnalysis

逃逸分析技术在 Java SE 6u23+ 开始支持,并默认设置为启用状态,可以不用额外加这个参数

public class JIT1
{
    /*169次循环后,消耗时间减少的原因:
    * 系统会在c2即时编译器上进行逃逸分析,发现 Object并没有在外围代码中使用,所以会自动略过这段代码。使消耗时间减少*/
    public static void main(String[] args) 
    {
        for (int i = 0; i < 200; i++) 
        {
            long start = System.nanoTime();
            for (int j = 0; j < 1000; j++) 
            {
                new Object(); //该对象创建完后并没有使用
            }
            long end = System.nanoTime();
            System.out.printf("%d\t%d\n",i,(end - start));
        }
    }
}
6.2. 方法内联
private static int square(final int i) 
{
    return i * i;
}
System.out.println(square(9));

如果发现 square 是热点方法,并且长度不太长时,会进行内联,所谓的内联就是把方法内代码拷贝、 粘贴到调用者的位置:

System.out.println(9 * 9);

还能够进行常量折叠(constant folding)的优化

System.out.println(8);

内联的例子

public class JIT2
{
    // -XX:+UnlockDiagnosticVMOptions -XX:+PrintInlining ;;;;打印内联
    // -XX:CompileCommand=dontinline,*JIT2.square ;;;;禁用square方法的内联
    // -XX:+PrintCompilation
    public static void main(String[] args)
    {
        int x = 0;
        for (int i = 0; i < 500; i++)
        {
            long start = System.nanoTime();
            for (int j = 0; j < 1000; j++)
            {
                x = square(9);

            }
            long end = System.nanoTime();
            System.out.printf("%d\t%d\t%d\n",i,x,(end - start)); //从300次循环后,所消耗的时间明显减少
        }
    }

    private static int square(final int i)
    {
        return i * i;
    }
}
6.3. 反射优化
public class Reflect1 
{
   public static void foo() 
   {
      System.out.println("foo...");
   }

   public static void main(String[] args) throws NoSuchMethodException, InvocationTargetException, IllegalAccessException, IOException
   {
      Method foo = Reflect1.class.getMethod("foo");
      for(int i = 0; i<=16; i++) 
      {
          System.out.printf("%d\t",i);
          foo.invoke(null);
      }
      System.in.read();
   }
}

foo.invoke 前面 0 ~ 15 次调用使用的是 MethodAccessor 的 NativeMethodAccessorImpl 实现

invoke方法源码;

@CallerSensitive
public Object invoke(Object obj, Object... args)
    throws IllegalAccessException, IllegalArgumentException,
       InvocationTargetException
{
    if (!override) {
        if (!Reflection.quickCheckMemberAccess(clazz, modifiers)) {
            Class<?> caller = Reflection.getCallerClass();
            checkAccess(caller, clazz, obj, modifiers);
        }
    }
    //MethodAccessor是一个接口,有3个实现类,其中有一个是抽象类
    MethodAccessor ma = methodAccessor;             // read volatile
    if (ma == null) {
        ma = acquireMethodAccessor();
    }
    return ma.invoke(obj, args);
}

img

会由DelegatingMehodAccessorImpl去调用NativeMethodAccessorImpl

NativeMethodAccessorImpl源码

class NativeMethodAccessorImpl extends MethodAccessorImpl {
    private final Method method;
    private DelegatingMethodAccessorImpl parent;
    private int numInvocations;

    NativeMethodAccessorImpl(Method var1) {
        this.method = var1;
    }
	
	//每次进行反射调用,会让numInvocation与ReflectionFactory.inflationThreshold的值(15)进行比较,并使使得numInvocation的值加一
	//如果numInvocation>ReflectionFactory.inflationThreshold,则会调用本地方法invoke0方法
    public Object invoke(Object var1, Object[] var2) throws IllegalArgumentException, InvocationTargetException {
        if (++this.numInvocations > ReflectionFactory.inflationThreshold() && !ReflectUtil.isVMAnonymousClass(this.method.getDeclaringClass())) {
            MethodAccessorImpl var3 = (MethodAccessorImpl)(new MethodAccessorGenerator()).generateMethod(this.method.getDeclaringClass(), this.method.getName(), this.method.getParameterTypes(), this.method.getReturnType(), this.method.getExceptionTypes(), this.method.getModifiers());
            this.parent.setDelegate(var3);
        }

        return invoke0(this.method, var1, var2);
    }

    void setParent(DelegatingMethodAccessorImpl var1) {
        this.parent = var1;
    }

    private static native Object invoke0(Method var0, Object var1, Object[] var2);
}
//ReflectionFactory.inflationThreshold()方法的返回值
private static int inflationThreshold = 15;
  • 一开始if条件不满足,就会调用本地方法invoke()
  • 随着numInvocation的增大,当它大于ReflectionFactory.inflationThreshold的值16时,就会将 本地方法访问器 替换为一个 运行时动态生成的访问器,来提高效率
    • 这时会从反射调用变为正常调用,即直接调用 Reflect1.foo()

img

五、JVM-JMM内存模型

  • 很多人将【java 内存结构】与【java 内存模型】傻傻分不清,【java 内存模型】是 Java Memory Model(JMM)的意思。
  • 简单的说,JMM 定义了一套在多线程读写共享数据时(成员变量、数组)时,对数据的可见性有序性、和原子性的规则和保障

1. 原子性

1-1 问题解析

提出问题:两个线程对初始值为 0 的静态变量一个做自增,一个做自减,各做 5000 次,结果是 0 吗?

public class Demo1 
{
    static int i = 0;

    public static void main(String[] args) throws InterruptedException 
    {
        Thread t1 = new Thread(() -> 
        {
            for (int j = 0; j < 50000; j++) 
            {
                i++;
            }
        });
        
        Thread t2 = new Thread(() -> 
        {
            for (int j = 0; j < 50000; j++) 
            {
                i--;
            }
        });
        
        t1.start();
        t2.start();
        t1.join();
        t2.join();
        System.out.println(i);
    }
}

以上的结果可能是正数、负数、零。为什么呢?因为 Java 中对静态变量的自增,自减不是原子操作

例如对于 i++ 而言(i 为静态变量),实际会产生如下的 JVM 字节码指令:

getstatic i // 获取静态变量i的值
iconst_1 	// 准备常量1
iadd 		// 加法
putstatic i // 将修改后的值存入静态变量i

而对应 i-- 也是类似:

getstatic i // 获取静态变量i的值
iconst_1 	// 准备常量1
isub 		// 减法
putstatic i // 将修改后的值存入静态变量i

而 Java 的内存模型如下,完成静态变量的自增,自减需要在主存线程内存中进行数据交换:

在这里插入图片描述

如果是单线程以上 8 行代码是顺序执行(不会交错)没有问题:

// 假设i的初始值为0
getstatic i // 线程1-获取静态变量i的值 线程内i=0
iconst_1 	// 线程1-准备常量1
iadd 		// 线程1-自增 线程内i=1
putstatic i // 线程1-将修改后的值存入静态变量i 静态变量i=1
getstatic i // 线程1-获取静态变量i的值 线程内i=1
iconst_1 	// 线程1-准备常量1
isub 		// 线程1-自减 线程内i=0
putstatic i // 线程1-将修改后的值存入静态变量i 静态变量i=0

但多线程下这 8 行代码可能交错运行(为什么会交错?思考一下):

出现负数的情况:

// 假设i的初始值为0
getstatic i // 线程1-获取静态变量i的值 线程内i=0
getstatic i // 线程2-获取静态变量i的值 线程内i=0
iconst_1 	// 线程1-准备常量1
iadd 		// 线程1-自增 线程内i=1
putstatic i // 线程1-将修改后的值存入静态变量i 静态变量i=1
iconst_1 	// 线程2-准备常量1
isub 		// 线程2-自减 线程内i=-1
putstatic i // 线程2-将修改后的值存入静态变量i 静态变量i=-1

出现正数的情况:

// 假设i的初始值为0
getstatic i // 线程1-获取静态变量i的值 线程内i=0
getstatic i // 线程2-获取静态变量i的值 线程内i=0
iconst_1 	// 线程1-准备常量1
iadd 		// 线程1-自增 线程内i=1
iconst_1 	// 线程2-准备常量1
isub 		// 线程2-自减 线程内i=-1
putstatic i // 线程2-将修改后的值存入静态变量i 静态变量i=-1
putstatic i // 线程1-将修改后的值存入静态变量i 静态变量i=1
1-2 解决方法

synchronized(同步关键字)

语法:

synchronized( 对象 ) 
{
    要作为原子操作代码
}

synchronized 解决并发问题:

public class Demo1 
{
    static int i = 0;
    static Object obj = new Object(); //对象锁:要求是个对象

    public static void main(String[] args) throws InterruptedException 
    {

        Thread t1 = new Thread(() -> 
        {
            for (int j = 0; j < 50000; j++) 
            {
                synchronized (obj) 
                {
                    i++;
                }

            }
        });
        
        Thread t2 = new Thread(() -> 
        {
            for (int j = 0; j < 50000; j++) 
            {
                synchronized (obj) 
                {
                    i--;
                }
            }
        });
        
        t1.start();
        t2.start();
        t1.join(); //join的作用:使t1运行完后,再去运行主线程
        t2.join();
        System.out.println(i);//输出为0
    }
}

为什么需要这里的 obj 对象呢?

我们可以这样理解:可以把 obj 想象成一个房间,线程 t1,t2 想象成两个人。

当线程 t1 执行到 synchronized(obj) 时就好比 t1 进入了这个房间,并反手锁住了门,在门内执行 count++ 代码。

这时候如果 t2 也运行到了 synchronized(obj) 时,它发现门被锁住了,只能在门外等待。

当 t1 执行完 synchronized{} 块内的代码,这时候才会解开门上的锁,从 obj 房间出来。t2 线程这时才可以进入 obj 房间,反锁住门,执行它的 count-- 代码。

怎么从JVM角度理解呢?(这里引用《Java并发编程的艺术》里的一段话)

从JVM规范中可以看到Synchonized在JVM里的实现原理,JVM基于进入和退出Monitor对象来实现方法同步和代码块同步,但两者的实现细节不一样。代码块同步是使用monitorenter 和monitorexit指令实现的。 monitorenter指令是在编译后插入到同步代码块的开始位置,而monitorexit是插入到方法结束处和异常处,JVM要保证每个monitorenter必须有对应的monitorexit与之配对。任何对象都有一个monitor与之关联,当且一个monitor被持有后,它将处于锁定状态。线程执行到monitorenter 指令时,将会尝试获取对象所对应的monitor的所有权,即尝试获得对象的锁。

2.可见性

2-1 退不出的循环

先来看一个现象,main 线程对 run 变量的修改对于 t 线程不可见,导致了 t 线程无法停止:

static boolean run = true;
public static void main(String[] args) throws InterruptedException 
{
    Thread t = new Thread(()->
    {
        while(run)
        {
            // ....
        }
    });
    
    t.start();
    Thread.sleep(1000);
    run = false; // 线程t不会如预想的停下来
}

为什么会这样?

  1. 初始状态, t 线程刚开始从主内存读取了 run 的值到工作内存。

在这里插入图片描述

  1. 因为 t 线程要频繁从主内存中读取 run 的值,JIT 编译器会将 run 的值缓存至自己工作内存中的高速缓存中,减少对主存中 run 的访问,提高效率

在这里插入图片描述

  1. 1 秒之后,main 线程修改了 run 的值,并同步至主存,而 t 是从自己工作内存中的高速缓存中读取这个变量的值,结果永远是旧值

在这里插入图片描述

2-2 解决办法

volatile(易变关键字)

它可以用来修饰成员变量静态成员变量,他可以避免线程从自己的工作缓存中查找变量的值,必须到主存中获取它的值,线程操作 volatile 变量都是直接操作主存,保证了共享变量的可见性,但不能保证原子性

public class Demo1 
{
    volatile static boolean run = true;

    public static void main(String[] args) throws InterruptedException 
    {
        Thread t = new Thread(() -> 
        {
            while (run) 
            {
				// ....
            }
        });
        
        t.start();
        Thread.sleep(1000);
        run = false; // 线程t不会如预想的停下来
    }
}

注意:

synchronized 语句块既可以保证代码块的原子性,也同时保证代码块内变量的可见性。但 缺点是synchronized是属于重量级操作,性能相对更低

如果在前面示例的死循环中加入 System.out.println() 会发现即使不加 volatile 修饰符,线程 t 也 能正确看到对 run 变量的修改了,想一想为什么?

原因,进入println源码

public void println(int x) 
{
    synchronized (this) 
    {
        print(x);
        newLine();
    }
}

可以看出加了synchronized,保证了每次run变量都会从主存中获取

3.有序性

3-1 诡异的结果

看下面一个栗子:

int num = 0;
boolean ready = false;
// 线程1 执行此方法
public void actor1(I_Result r) 
{
    if(ready) 
    {
        r.r1 = num + num;
    } 
    else 
    {
        r.r1 = 1;
    }
}

// 线程2 执行此方法
public void actor2(I_Result r) 
{
    num = 2;
    ready = true;
}

看到这里可能聪明的小伙伴会想到有下面三种情况:

情况1:线程1 先执行,这时 ready = false,所以进入 else 分支结果为 1

情况2:线程2 先执行 num = 2,但没来得及执行 ready = true,线程1 执行,还是进入 else 分支,结果为1

情况3:线程2 执行到 ready = true,线程1 执行,这回进入 if 分支,结果为 4(因为 num 已经执行过了)

但其实还有可能为0哦!😲

有可能还是:线程 2 执行 ready=true ,切换到线程1 ,进入if分支,相加为0,再切回线程 2 执行 num=2

这种现象就是指令重排,是JIT编译器在运行时的一些优化,这个现象需要通过大量测试才能复现

3-2 解决方法

volatile 修饰的变量,可以禁用指令重排

//下方出现的所有注解不用在意
@JCStressTest
@Outcome(id = {"1", "4"}, expect = Expect.ACCEPTABLE, desc = "ok")
@Outcome(id = "0", expect = Expect.ACCEPTABLE_INTERESTING, desc = "!!!!")
@State
public class ConcurrencyTest 
{
    int num = 0;
    volatile boolean ready = false;//可以禁用指令重排
    @Actor
    public void actor1(I_Result r) 
    {
        if(ready) 
        {
            r.r1 = num + num;
        } 
        else 
        {
            r.r1 = 1;
        }
    }
    
    @Actor
    public void actor2(I_Result r) 
    {
        num = 2;
        ready = true;
    }
}
3-3 有序性理解

同一线程内,JVM会在不影响正确性的前提下,可以调整语句的执行顺序,看看下面的代码:

static int i;
static int j;
// 在某个线程内执行如下赋值操作
i = ...; // 较为耗时的操作
j = ...;

可以看到,至于是先执行 i 还是 先执行 j ,对最终的结果不会产生影响。所以,上面代码真正执行时, 既可以是

i = ...; // 较为耗时的操作
j = ...;

也可以是

j = ...;
i = ...; // 较为耗时的操作

这种特性称之为指令重排多线程下指令重排会影响正确性,例如著名的 double-checked locking 模式实现单例

public class Singleton 
{
    private Singleton(){}

    private static Singleton INSTANCE = null;

    public static Singleton getInstance() 
    {
        //步骤1: 实例没创建,才会进入内部的 synchronized 代码块
        if (INSTANCE == null) 
        {
            //步骤2: 可能第一个线程在synchronized 代码块还没创建完对象时,第二个线程已经到了这一步,所以里面还需要加上判断
            synchronized (Singleton.class) 
            {
                //步骤3: 也许有其他线程已经创建实例,所以再判断一次
                if (INSTANCE == null) 
                {
                    INSTANCE = new Singleton();
                }
            }
        }
        return INSTANCE;
    }
}

//double check 的原因:线程1进入到步骤3,但是线程1还在创建Singleton对象;此时线程2在步骤1判断为null,进入到步骤2阻塞;当线程1创建完对象后,线程2进入步骤2,此时如果没有步骤3,线程2会再次创建一遍Singleton对象。所以需要两次非空判断。

以上的实现特点是:

  • 懒惰实例化
  • 首次使用 getInstance() 才使用 synchronized 加锁,后续使用时无需加锁

上面的代码看似已经很完美了,但是在多线程环境下还是会有指令重排问题

INSTANCE = new Singleton() 对应的字节码为:

0: new #2 			// class cn/itcast/jvm/t4/Singleton
3: dup
4: invokespecial #3 // Method "<init>":()V
7: putstatic #4 	// Field INSTANCE:Lcn/itcast/jvm/t4/Singleton;

其中4、7 两步顺序不是固定的,也许 jvm 会优化为:先将引用地址赋值给 INSTANCE 变量后,再执行构造方法,如果两个线程 t1,t2 按如下时间顺序执行:

时间1 t1 线程执行到 INSTANCE = new Singleton();
时间2 t1 线程分配空间,为Singleton对象生成了引用地址(第0行)
时间3 t1 线程将引用地址赋值给 INSTANCE,这时 INSTANCE != null(第7行)
时间4 t2 线程进入getInstance() 方法,发现 INSTANCE != nullsynchronized块外),直接返回 INSTANCE
时间5 t1 线程执行Singleton的构造方法(第4行)

这时 t1 还未完全将构造方法执行完毕,如果在构造方法中要执行很多初始化操作,那么 t2 拿到的是将 是一个未初始化完毕的单例

对 INSTANCE 使用 volatile 修饰即可,可以禁用指令重排,但要注意在 JDK 5 以上的版本的 volatile 才 会真正有效

3-4 happens-before

happens-before 规定了哪些写操作对其它线程的读操作可见,它是可见性与有序性的一套规则总结,抛开以下 happens-before 规则,JMM 并不能保证一个线程对共享变量的写,对于其它线程对该共享变量的读可见

  • 线程解锁 m 之前对变量的写,对于接下来对 m 加锁的其它线程对该变量的读可见
static int x;
static Object m = new Object();
new Thread(()->
{
    synchronized(m) 
    {
        x = 10;
    }
},"t1").start();

new Thread(()->
{
    synchronized(m) 
    {
        System.out.println(x);
    }
},"t2").start()
  • 线程对 volatile 变量的写,对接下来其它线程对该变量的读可见
volatile static int x;
new Thread(()->
{
    x = 10;
},"t1").start();

new Thread(()->
{
    System.out.println(x);
},"t2").start();
  • 线程 start 前对变量的写,对该线程开始后对该变量的读可见
static int x;
x = 10;

new Thread(()->
{
    System.out.println(x);
},"t2").start();
  • 线程结束前对变量的写,对其它线程得知它结束后的读可见(比如其它线程调用 t1.isAlive()t1.join()等待它结束)
static int x;
Thread t1 = new Thread(()->
{
    x = 10;
},"t1");

t1.start();
t1.join();
System.out.println(x);
  • 线程 t1 打断 t2(interrupt)前对变量的写,对于其他线程得知 t2 被打断后对变量的读可见(通 过t2.interruptedt2.isInterrupted
static int x;
public static void main(String[] args) 
{
    Thread t2 = new Thread(()->
    {
        while(true) 
        {
            if(Thread.currentThread().isInterrupted()) 
            {
                System.out.println(x);//10
                break;
            }
        }
    },"t2");
    
    t2.start();
    new Thread(()->
    {
        try 
        {
            Thread.sleep(1000);
        } 
        catch (InterruptedException e) 
        {
            e.printStackTrace();
        }
        x = 10; //线程 t1 打断 t2(interrupt)前对变量的写
        t2.interrupt(); //打断t2线程
    },"t1").start();
    
    while(!t2.isInterrupted())  //主线程等待t2被打断
    {
        Thread.yield();
    }
    System.out.println(x); //10;主线程得知 t2 被打断后对变量的读可见
}
  • 对变量默认值(0,false,null)的写,对其它线程对该变量的读可见
  • 具有传递性,如果 x hb-> y 并且 y hb-> z 那么有 x hb-> z

以上变量都是指共享变量 (即成员变量或静态资源变量)

4.CAS与原子类

4-1 CAS

CASCompare and Swap ,它体现的一种乐观锁的思想

比如多个线程要对一个共享的整型变量执行 +1 操作:

// 需要不断尝试
while(true) 
{
    int 旧值 = 共享变量 ; // 比如拿到了当前值 0
    int 结果 = 旧值 + 1; // 在旧值 0 的基础上增加 1 ,正确结果是 1
    /*
	这时候如果别的线程把共享变量改成了 5,本线程的正确结果 1 就作废了,这时候
	compareAndSwap 返回 false,重新尝试,直到:
	compareAndSwap 返回 true,表示我本线程做修改的同时,别的线程没有干扰
	*/
    if( compareAndSwap ( 旧值, 结果 )) 
    {
        // 旧值和结果分别都一样,成功,退出循环
    }
    //不一样,继续循环尝试
}

获取共享变量时,为了保证该变量的可见性,需要使用 volatile 修饰。结合 CAS 和 volatile 可以实现无锁并发,适用于竞争不激烈、多核 CPU 的场景下。

  • 因为没有使用 synchronized,所以线程不会陷入阻塞,这是效率提升的因素之一

  • 但如果竞争激烈,可以想到重试必然频繁发生,反而效率会受影响

CAS 底层依赖于一个 Unsafe 类来直接调用操作系统底层的 CAS 指令,下面是直接使用 Unsafe 对象进行线程安全保护的一个例子:

public class TestCAS
{
    public static void main(String[] args) throws InterruptedException 
    {
        DataContainer dc = new DataContainer();
        int count = 5;
        Thread t = new Thread(() -> 
        {
            for (int i = 0; i < count; i++) 
            {
                dc.increase();
            }
        });
        t.start();
        t.join();
        System.out.println(dc.getData());
    }
}

//我们通过CAS的思想实现了一个无锁并发的 对整数共享数据的保护
class DataContainer 
{
    private volatile int data;
    static final Unsafe unsafe;
    static final long DATA_OFFSET; //数据偏移量

    static 
    {
        try 
        {
            // Unsafe 对象不能直接调用,只能通过反射获得
            Field theUnsafe = Unsafe.class.getDeclaredField("theUnsafe");
            theUnsafe.setAccessible(true);
            unsafe = (Unsafe) theUnsafe.get(null);
        } 
        catch (NoSuchFieldException | IllegalAccessException e) 
        {
            throw new Error(e);
        }
        try 
        {
            // data 属性在 DataContainer 对象中的偏移量,用于 Unsafe 直接访问该属性
            DATA_OFFSET = unsafe.objectFieldOffset(DataContainer.class.getDeclaredField("data"));
        } 
        catch (NoSuchFieldException e) 
        {
            throw new Error(e);
        }
    }

    //---------------写操作---------------------
    public void increase() 
    {
        int oldValue;
        while (true) 
        {
            // 获取共享变量旧值,可以在这一行加入断点,修改 data 调试来加深理解
            oldValue = data;
            // cas 尝试修改 data 为 旧值 + 1,如果期间旧值被别的线程改了,返回 false
            if (unsafe.compareAndSwapInt(this, DATA_OFFSET, oldValue, oldValue + 1)) 
            {
                return;
            }
        }
    }

    public void decrease() 
    {
        int oldValue;
        while (true) 
        {
            oldValue = data;
            if (unsafe.compareAndSwapInt(this, DATA_OFFSET, oldValue, oldValue - 1)) 
            {
                return;
            }
        }
    }

    //---------------读操作---------------------
    public int getData() 
    {
        return data;
    }
}
4-2 乐观锁与悲观锁
  • CAS 是基于乐观锁的思想:最乐观的估计,不怕别的线程来修改共享变量,就算改了也没关系,我吃亏点再重试呗。
  • synchronized 是基于悲观锁的思想:最悲观的估计,得防着其它线程来修改共享变量,我上了锁你们都别想改,我改完了解开锁,你们才有机会
4-3 原子操作类

juc(java.util.concurrent)中提供了原子操作类,可以提供线程安全的操作,例如:AtomicInteger、 AtomicBoolean等,它们底层就是采用 CAS 技术 + volatile 来实现的。 可以使用 AtomicInteger 改写之前的例子:

public class TestCAS 
{
    //创建原子整数对象
    private static AtomicInteger i = new AtomicInteger(0);

    public static void main(String[] args) throws InterruptedException 
    {
        Thread t1 = new Thread(() -> 
        {
            for (int j = 0; j < 5000; j++) 
            {
                i.getAndIncrement(); //获取并且自增 i++
            }
        });
        
        Thread t2 = new Thread(() -> 
        {
            for (int j = 0; j < 5000; j++) 
            {
                i.getAndDecrement(); //获取并且自减 i--
            }
        });
        
        t1.start();
        t2.start();
        t1.join();
        t2.join();
        System.out.println(i);//0
    }
}

5.synchronized 优化

Java HotSpot 虚拟机中,每个对象都有对象头(包括 class 指针和 Mark Word)。Mark Word 平时存储这个对象的哈希码 、 分代年龄 ,当加锁时,这些信息就根据情况被替换为标记位 、 线程锁记录指针 、 重量级锁指针 、 线程ID 等内容

5-1 轻量级锁

如果一个对象虽然有多线程访问,但多线程访问的时间是错开的(也就是没有竞争),那么可以使用轻量级锁来优化。这就好比:

学生(线程 A)用课本占座,上了半节课,出门了(CPU时间到),回来一看,发现课本没变,说明有竞争,继续上他的课。 如果这期间有其它学生(线程 B)来了,会告知(线程A)有并发访问,线程A 随即升级为重量级锁,进入重量级锁的流程。

而重量级锁就不是那么用课本占座那么简单了,可以想象线程 A 走之前,把座位用一个铁栅栏围起来,假设有两个方法同步块,利用同一个对象加锁

static Object obj = new Object();
public static void method1() 
{
    synchronized( obj ) 
    {
        // 同步块 A
        method2();
    }
}
public static void method2() 
{
    synchronized( obj ) 
    {
        // 同步块 B
    }
}

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-GQ3rU9YA-1669108012115)(C:\Users\vanpersie\AppData\Roaming\Typora\typora-user-images\image-20221122162721712.png)]

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-bpDIm7RM-1669108012115)(C:\Users\vanpersie\AppData\Roaming\Typora\typora-user-images\image-20221122163135177.png)]

5-2 锁膨胀

如果在尝试加轻量级锁的过程中,CAS 操作无法成功,这时一种情况就是有其它线程为此对象加上了轻量级锁(有竞争),这时需要进行锁膨胀,将轻量级锁变为重量级锁。

static Object obj = new Object();
public static void method1() 
{
    synchronized( obj ) 
    {
        // 同步块 
    }
}

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-jI88Y6bN-1669108012115)(C:\Users\vanpersie\AppData\Roaming\Typora\typora-user-images\image-20221122163444814.png)]

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-DUp6midu-1669108012115)(C:\Users\vanpersie\AppData\Roaming\Typora\typora-user-images\image-20221122163644208.png)]

5-3 重量锁

重量级锁竞争的时候,还可以使用自旋来进行优化,如果当前线程自旋成功(即这时候持锁线程已经退出了同步块,释放了锁),这时当前线程就可以避免阻塞。

在 Java 6 之后自旋锁是自适应的,比如对象刚刚的一次自旋操作成功过,那么认为这次自旋成功的可能性会高,就多自旋几次;反之,就少自旋甚至不自旋,总之,比较智能。

  • 自旋会占用CPU时间,单核CPU自旋就是浪费,多核CPU自旋才能发挥优势

  • 好比等红灯时汽车是不是熄火,不熄火相当于自旋(等待时间短了划算),熄火了相当于阻塞(等待时间长了划算)

  • Java 7 之后不能控制是否开启自旋功能

自旋重试成功的情况

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-p1RwocZS-1669108012116)(C:\Users\vanpersie\AppData\Roaming\Typora\typora-user-images\image-20221122165011150.png)]

自旋重试失败的情况

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-oO0G6104-1669108012116)(C:\Users\vanpersie\AppData\Roaming\Typora\typora-user-images\image-20221122165103468.png)]

5-4 偏向锁

轻量级锁在没有竞争时(就自己这个线程),每次重入仍然需要执行CAS操作。Java 6中引入了偏向锁来做进一步优化:只有第一次使用CAS将线程ID设置到对象的Mark Word头,之后发现这个线程ID是自己的就表示没有竞争,不用重新CAS

  • 撤销偏向需要将持锁线程升级为轻量级锁,这个过程中所有线程需要暂停(STW)

  • 访问对象的 hashCode 也会撤销偏向锁

  • 如果对象虽然被多个线程访问,但没有竞争,这时偏向了线程 T1 的对象仍有机会重新偏向 T2,重偏向会重置对象的 Thread ID

  • 撤销偏向和重偏向都是批量进行的,以类为单位

  • 如果撤销偏向到达某个阈值,整个类的所有对象都会变为不可偏向的

  • 可以主动使用 -XX:-UseBiasedLocking 禁用偏向锁

    假设有两个方法同步块,利用同一个对象加锁

static Object obj = new Object();
public static void method1() 
{
    synchronized( obj ) 
    {
        // 同步块 A
        method2();
    }
}
public static void method2() 
{
    synchronized( obj ) 
    {
        // 同步块 B
    }
}

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-wApR2aib-1669108012116)(C:\Users\vanpersie\AppData\Roaming\Typora\typora-user-images\image-20221122165844599.png)]

5-5 其他优化
(1). 减少上锁时间

同步代码块中尽量短

(2). 减少锁的粒度

将一个锁拆分为多个锁提高并发度,例如:

  • ConcurrentHashMap

  • LongAdder 分为 base 和 cells 两部分。没有并发争用的时候或者是 cells 数组正在初始化的时候,会使用 CAS 来累加到base,有并发争用,会初始化 cells 数组,数组有多少个 cell,就允许有多少线程并行修改,最后将数组中每个 cell 累加,再加上 base 就是最终的值

  • LinkedBlockingQueue 入队和出队使用不同的锁,相对于LinkedBlockingArray只有一个锁效率要高

(3). 锁粗化

多次循环进入同步块不如同步块内多次循环,另外 JVM 可能会做如下优化,把多次 append 的加锁操作粗化为一次(因为都是对同一个对象加锁,没必要重入多次)

new StringBuffer().append("a").append("b").append("c");
(4). 锁消除

JVM 会进行代码的逃逸分析,例如某个加锁对象是方法内局部变量,不会被其它线程所访问到,这时候就会被即时编译器忽略掉所有同步操作

(5). 读写分离

如果撤销偏向到达某个阈值,整个类的所有对象都会变为不可偏向的

ompare and Swap` ,它体现的一种乐观锁的思想

比如多个线程要对一个共享的整型变量执行 +1 操作:

// 需要不断尝试
while(true) 
{
    int 旧值 = 共享变量 ; // 比如拿到了当前值 0
    int 结果 = 旧值 + 1; // 在旧值 0 的基础上增加 1 ,正确结果是 1
    /*
	这时候如果别的线程把共享变量改成了 5,本线程的正确结果 1 就作废了,这时候
	compareAndSwap 返回 false,重新尝试,直到:
	compareAndSwap 返回 true,表示我本线程做修改的同时,别的线程没有干扰
	*/
    if( compareAndSwap ( 旧值, 结果 )) 
    {
        // 旧值和结果分别都一样,成功,退出循环
    }
    //不一样,继续循环尝试
}

获取共享变量时,为了保证该变量的可见性,需要使用 volatile 修饰。结合 CAS 和 volatile 可以实现无锁并发,适用于竞争不激烈、多核 CPU 的场景下。

  • 因为没有使用 synchronized,所以线程不会陷入阻塞,这是效率提升的因素之一

  • 但如果竞争激烈,可以想到重试必然频繁发生,反而效率会受影响

CAS 底层依赖于一个 Unsafe 类来直接调用操作系统底层的 CAS 指令,下面是直接使用 Unsafe 对象进行线程安全保护的一个例子:

public class TestCAS
{
    public static void main(String[] args) throws InterruptedException 
    {
        DataContainer dc = new DataContainer();
        int count = 5;
        Thread t = new Thread(() -> 
        {
            for (int i = 0; i < count; i++) 
            {
                dc.increase();
            }
        });
        t.start();
        t.join();
        System.out.println(dc.getData());
    }
}

//我们通过CAS的思想实现了一个无锁并发的 对整数共享数据的保护
class DataContainer 
{
    private volatile int data;
    static final Unsafe unsafe;
    static final long DATA_OFFSET; //数据偏移量

    static 
    {
        try 
        {
            // Unsafe 对象不能直接调用,只能通过反射获得
            Field theUnsafe = Unsafe.class.getDeclaredField("theUnsafe");
            theUnsafe.setAccessible(true);
            unsafe = (Unsafe) theUnsafe.get(null);
        } 
        catch (NoSuchFieldException | IllegalAccessException e) 
        {
            throw new Error(e);
        }
        try 
        {
            // data 属性在 DataContainer 对象中的偏移量,用于 Unsafe 直接访问该属性
            DATA_OFFSET = unsafe.objectFieldOffset(DataContainer.class.getDeclaredField("data"));
        } 
        catch (NoSuchFieldException e) 
        {
            throw new Error(e);
        }
    }

    //---------------写操作---------------------
    public void increase() 
    {
        int oldValue;
        while (true) 
        {
            // 获取共享变量旧值,可以在这一行加入断点,修改 data 调试来加深理解
            oldValue = data;
            // cas 尝试修改 data 为 旧值 + 1,如果期间旧值被别的线程改了,返回 false
            if (unsafe.compareAndSwapInt(this, DATA_OFFSET, oldValue, oldValue + 1)) 
            {
                return;
            }
        }
    }

    public void decrease() 
    {
        int oldValue;
        while (true) 
        {
            oldValue = data;
            if (unsafe.compareAndSwapInt(this, DATA_OFFSET, oldValue, oldValue - 1)) 
            {
                return;
            }
        }
    }

    //---------------读操作---------------------
    public int getData() 
    {
        return data;
    }
}
4-2 乐观锁与悲观锁
  • CAS 是基于乐观锁的思想:最乐观的估计,不怕别的线程来修改共享变量,就算改了也没关系,我吃亏点再重试呗。
  • synchronized 是基于悲观锁的思想:最悲观的估计,得防着其它线程来修改共享变量,我上了锁你们都别想改,我改完了解开锁,你们才有机会
4-3 原子操作类

juc(java.util.concurrent)中提供了原子操作类,可以提供线程安全的操作,例如:AtomicInteger、 AtomicBoolean等,它们底层就是采用 CAS 技术 + volatile 来实现的。 可以使用 AtomicInteger 改写之前的例子:

public class TestCAS 
{
    //创建原子整数对象
    private static AtomicInteger i = new AtomicInteger(0);

    public static void main(String[] args) throws InterruptedException 
    {
        Thread t1 = new Thread(() -> 
        {
            for (int j = 0; j < 5000; j++) 
            {
                i.getAndIncrement(); //获取并且自增 i++
            }
        });
        
        Thread t2 = new Thread(() -> 
        {
            for (int j = 0; j < 5000; j++) 
            {
                i.getAndDecrement(); //获取并且自减 i--
            }
        });
        
        t1.start();
        t2.start();
        t1.join();
        t2.join();
        System.out.println(i);//0
    }
}

5.synchronized 优化

Java HotSpot 虚拟机中,每个对象都有对象头(包括 class 指针和 Mark Word)。Mark Word 平时存储这个对象的哈希码 、 分代年龄 ,当加锁时,这些信息就根据情况被替换为标记位 、 线程锁记录指针 、 重量级锁指针 、 线程ID 等内容

5-1 轻量级锁

如果一个对象虽然有多线程访问,但多线程访问的时间是错开的(也就是没有竞争),那么可以使用轻量级锁来优化。这就好比:

学生(线程 A)用课本占座,上了半节课,出门了(CPU时间到),回来一看,发现课本没变,说明有竞争,继续上他的课。 如果这期间有其它学生(线程 B)来了,会告知(线程A)有并发访问,线程A 随即升级为重量级锁,进入重量级锁的流程。

而重量级锁就不是那么用课本占座那么简单了,可以想象线程 A 走之前,把座位用一个铁栅栏围起来,假设有两个方法同步块,利用同一个对象加锁

static Object obj = new Object();
public static void method1() 
{
    synchronized( obj ) 
    {
        // 同步块 A
        method2();
    }
}
public static void method2() 
{
    synchronized( obj ) 
    {
        // 同步块 B
    }
}

[外链图片转存中…(img-GQ3rU9YA-1669108012115)]

[外链图片转存中…(img-bpDIm7RM-1669108012115)]

5-2 锁膨胀

如果在尝试加轻量级锁的过程中,CAS 操作无法成功,这时一种情况就是有其它线程为此对象加上了轻量级锁(有竞争),这时需要进行锁膨胀,将轻量级锁变为重量级锁。

static Object obj = new Object();
public static void method1() 
{
    synchronized( obj ) 
    {
        // 同步块 
    }
}

[外链图片转存中…(img-jI88Y6bN-1669108012115)]

[外链图片转存中…(img-DUp6midu-1669108012115)]

5-3 重量锁

重量级锁竞争的时候,还可以使用自旋来进行优化,如果当前线程自旋成功(即这时候持锁线程已经退出了同步块,释放了锁),这时当前线程就可以避免阻塞。

在 Java 6 之后自旋锁是自适应的,比如对象刚刚的一次自旋操作成功过,那么认为这次自旋成功的可能性会高,就多自旋几次;反之,就少自旋甚至不自旋,总之,比较智能。

  • 自旋会占用CPU时间,单核CPU自旋就是浪费,多核CPU自旋才能发挥优势

  • 好比等红灯时汽车是不是熄火,不熄火相当于自旋(等待时间短了划算),熄火了相当于阻塞(等待时间长了划算)

  • Java 7 之后不能控制是否开启自旋功能

自旋重试成功的情况

[外链图片转存中…(img-p1RwocZS-1669108012116)]

自旋重试失败的情况

[外链图片转存中…(img-oO0G6104-1669108012116)]

5-4 偏向锁

轻量级锁在没有竞争时(就自己这个线程),每次重入仍然需要执行CAS操作。Java 6中引入了偏向锁来做进一步优化:只有第一次使用CAS将线程ID设置到对象的Mark Word头,之后发现这个线程ID是自己的就表示没有竞争,不用重新CAS

  • 撤销偏向需要将持锁线程升级为轻量级锁,这个过程中所有线程需要暂停(STW)

  • 访问对象的 hashCode 也会撤销偏向锁

  • 如果对象虽然被多个线程访问,但没有竞争,这时偏向了线程 T1 的对象仍有机会重新偏向 T2,重偏向会重置对象的 Thread ID

  • 撤销偏向和重偏向都是批量进行的,以类为单位

  • 如果撤销偏向到达某个阈值,整个类的所有对象都会变为不可偏向的

  • 可以主动使用 -XX:-UseBiasedLocking 禁用偏向锁

    假设有两个方法同步块,利用同一个对象加锁

static Object obj = new Object();
public static void method1() 
{
    synchronized( obj ) 
    {
        // 同步块 A
        method2();
    }
}
public static void method2() 
{
    synchronized( obj ) 
    {
        // 同步块 B
    }
}

[外链图片转存中…(img-wApR2aib-1669108012116)]

5-5 其他优化
(1). 减少上锁时间

同步代码块中尽量短

(2). 减少锁的粒度

将一个锁拆分为多个锁提高并发度,例如:

  • ConcurrentHashMap

  • LongAdder 分为 base 和 cells 两部分。没有并发争用的时候或者是 cells 数组正在初始化的时候,会使用 CAS 来累加到base,有并发争用,会初始化 cells 数组,数组有多少个 cell,就允许有多少线程并行修改,最后将数组中每个 cell 累加,再加上 base 就是最终的值

  • LinkedBlockingQueue 入队和出队使用不同的锁,相对于LinkedBlockingArray只有一个锁效率要高

(3). 锁粗化

多次循环进入同步块不如同步块内多次循环,另外 JVM 可能会做如下优化,把多次 append 的加锁操作粗化为一次(因为都是对同一个对象加锁,没必要重入多次)

new StringBuffer().append("a").append("b").append("c");
(4). 锁消除

JVM 会进行代码的逃逸分析,例如某个加锁对象是方法内局部变量,不会被其它线程所访问到,这时候就会被即时编译器忽略掉所有同步操作

(5). 读写分离

如果撤销偏向到达某个阈值,整个类的所有对象都会变为不可偏向的

  • 可以主动使用 -XX:-UseBiasedLocking 禁用偏向锁
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值