JVM工作原理与实战(二十二):方法区的垃圾回收

简介: JVM作为Java程序的运行环境,其负责解释和执行字节码,管理内存,确保安全,支持多线程和提供性能监控工具,以及确保程序的跨平台运行。本文主要介绍了Java的内存管理和自动垃圾回收、方法区的垃圾回收等内容。

一、Java的内存管理和自动垃圾回收

运行时数据区知识回顾

Java虚拟机(JVM)在运行Java程序期间,会创建并维护一系列内存区域,这些区域总称为运行时数据区。这些区域根据其用途和特性,被严格定义并管理。《Java虚拟机规范》详细规定了这些区域的作用和行为,以确保所有Java虚拟机实现的一致性和正确性。

线程不共享区域:

  • 程序计数器:用于存储当前线程执行的字节码指令地址。这个区域是每个线程独有的,不共享。
  • Java虚拟机栈:每个线程在创建时都会创建一个虚拟机栈,每个方法调用都会创建一个栈帧,用于存储局部变量、操作数栈、动态链接和方法出口信息。
  • 本地方法栈:与虚拟机栈相似,本地方法栈为native方法提供服务。

线程共享区域:

  • 方法区:用于存储已被JVM加载的类信息、常量、静态变量以及即时编译器编译后的代码等数据。
  • 堆:堆是所有线程共享的区域,用于动态分配内存。所有的对象实例以及数组都应当在堆上分配。

image.gif

在运行时数据区中,线程不共享的部分都是伴随着线程的创建而创建,随着线程的销毁而销毁。而方法的栈帧在执行完方法之后就会自动弹出栈并释放掉对应的内存。这种内存管理方式可以有效地减少内存的浪费,并且可以防止内存泄漏问题的发生。

image.gif

二、方法区的垃圾回收

1.回收条件

在Java中,方法区用于存储已被虚拟机加载的类信息、常量、静态变量以及即时编译器编译后的代码等数据。与堆区一样,方法区也有垃圾回收机制,主要是对不再使用的类进行回收。

要判定一个类是否可以被卸载,需要同时满足以下三个条件:

  • 此类所有的实例对象都已经被回收,即在堆区中不存在该类的任何实例对象和子类对象。
  • 加载该类的类加载器已经被回收。在Java中,类加载器通过ClassLoader类实现,当一个类加载器不再被引用,或者其父类加载器也被回收时,该类加载器会被回收。
  • 该类对应的java.lang.Class对象没有被其他对象引用。如果一个Class对象没有被其他任何对象引用,那么这个Class对象可以被回收。

2.手动触发垃圾回收

虽然垃圾回收是自动进行的,但在某些情况下,可能需要手动触发垃圾回收。这可以通过调用System.gc()方法实现。这个方法会向Java虚拟机发送一个请求,要求进行垃圾回收。

System.gc();

image.gif

需要注意的是,调用System.gc()并不一定会立即执行垃圾回收。Java虚拟机可能会根据当前的运行情况自行判断是否需要进行垃圾回收。因此,手动触发垃圾回收并不一定能够提高程序的性能。

在开发中,手动触发垃圾回收的情况并不常见。但在一些特殊的应用场景中,如OSGi框架和JSP的热部署等,可能需要手动触发垃圾回收。在这些场景中,当一个jsp文件被修改后,对应的类加载器会被卸载,然后重新创建类加载器并重新加载jsp文件。

3.方法区的垃圾回收案例

案例:

此类所有的实例对象都已经被回收,即在堆区中不存在该类的任何实例对象和子类对象。

public class Demo1 {
    public static void main(String[] args) throws InterruptedException {
        try {
            URLClassLoader loader = new URLClassLoader(new URL[]{new URL("file:D:\\Test\\")});
            Class<?> clazz = loader.loadClass("com.rye.test.Test");
            Object o = clazz.newInstance();
            o = null;
            System.gc();
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}
image.gif

image.gif

案例:

加载该类的类加载器已经被回收。在Java中,类加载器通过ClassLoader类实现,当一个类加载器不再被引用,或者其父类加载器也被回收时,该类加载器会被回收。

public class Demo1 {
    public static void main(String[] args) throws InterruptedException {
        try {
            URLClassLoader loader = new URLClassLoader(new URL[]{new URL("file:D:\\Test\\")});
            Class<?> clazz = loader.loadClass("com.rye.test.Test");
            Object o = clazz.newInstance();
            loader = null;
            System.gc();
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}
image.gif

image.gif

案例:

该类对应的java.lang.Class对象没有被其他对象引用。如果一个Class对象没有被其他任何对象引用,那么这个Class对象可以被回收。

public class Demo1 {
    public static void main(String[] args) throws InterruptedException {
        try {
            URLClassLoader loader = new URLClassLoader(new URL[]{new URL("file:D:\\Test\\")});
            Class<?> clazz = loader.loadClass("com.rye.test.Test");
            Object o = clazz.newInstance();
            clazz = null;
            System.gc();
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}
image.gif

image.gif

总结

JVM是Java程序的运行环境,负责字节码解释、内存管理、安全保障、多线程支持、性能监控和跨平台运行。本文主要介绍了Java的内存管理和自动垃圾回收、方法区的垃圾回收等内容,希望对大家有所帮助。

相关文章
|
1月前
|
Oracle Java 关系型数据库
JVM深入原理(一+二):JVM概述和JVM功能
JVM全称是Java Virtual Machine-Java虚拟机JVM作用:本质上是一个运行在计算机上的程序,职责是运行Java字节码文件,编译为机器码交由计算机运行。
59 0
|
1月前
|
Arthas 存储 Java
JVM深入原理(三+四):JVM组成和JVM字节码文件
目录3. JVM组成3.1. 组成-运行时数据区3.2. 组成-类加载器3.3. 组成-执行引擎3.4. 组成-本地接口4. JVM字节码文件4.1. 字节码文件-组成4.1.1. 组成-基础信息4.1.1.1. 基础信息-魔数4.1.1.2. 基础信息-主副版本号4.1.2. 组成-常量池4.1.3. 组成-方法4.1.3.1. 方法-工作流程4.1.4. 组成-字段4.1.5. 组成-属性4.2. 字节码文件-查看工具4.2.1. javap4.2.2. jclasslib4.2.3. 阿里Arthas
36 0
|
1月前
|
存储 安全 Java
JVM深入原理(五):JVM组成和JVM字节码文件
类的生命周期概述:类的生命周期描述了一个类加载,使用,卸载的整个过类的生命周期阶段:类的声明周期主要分为五个阶段:加载->连接->初始化->使用->卸载,其中连接中分为三个小阶段验证->准备->解析。
27 0
|
1月前
|
Arthas Java 测试技术
JVM深入原理(六)(一):JVM类加载器
目录6. JVM类加载器6.1. 类加载器-概述6.2. 类加载器-执行流程6.3. 类加载器-分类(JDK8)6.3.1. JVM底层实现的类加载器6.3.1.1. 启动类加载器6.3.2. Java代码实现类的加载器6.3.2.1. 扩展类加载器6.3.2.2. 应用程序类加载器6.4. 类加载器-Arthas查看类加载器
29 0
|
1月前
|
Java 关系型数据库 MySQL
JVM深入原理(六)(二):双亲委派机制
自定义类加载器打破双亲委派机制的方法:复写ClassLoader中的loadClass方法常见问题:要加载的类名如果是以java.开头,则会抛出安全性异常加载自定义的类都会有一个共同的父类Object,需要在代码中交由父类加载器去加载自定义类加载器不手动指定parent会默认指定应用类加载两个自定义类加载器加载同一个类会被认为是两个对象,只有相同的类加载器+想通的类限定名才会被认为是一个对象。
39 0
|
1月前
|
存储 安全 Java
JVM深入原理(七)(一):运行时数据区
栈的介绍:Java虚拟机栈采用栈的数据结构来管理方法调用中的基本数据,先进后出,每一个方法的调用使用一个栈帧来保存栈的组成:栈:一个线程运行所需要的内存空间,一个栈由多个栈帧组成栈帧:一个方法运行所需要的内存空间活动栈帧:一个线程中只能有一个活动栈帧栈的生命周期:栈随着线程的创建而创建,而回收会在线程销毁时进行栈的执行流程:栈帧压入栈内执行方法执行完毕释放内存若方法间存在调用,那么会压入被调用方法入栈,执行完后释放内存,再执行当前方法,直到执行完毕,释放所有内存。
30 0
|
1月前
|
存储 缓存 安全
JVM深入原理(七)(二):运行时数据区
堆的作用:存放对象的内存空间,它是空间最大的一块内存区域.栈上的局部变量表中,可以存放堆上对象的引用。静态变量也可以存放堆对象的引用,通过静态变量就可以实现对象在线程之间共享。堆的特点:线程共享:堆中的对象都需要考虑线程安全的问题垃圾回收:堆有垃圾回收机制,不再引用的对象就会被回收方法区的概述:方法区是存放基础信息的位置,线程共享,主要包括:类的元信息:保存了所有类的基本信息运行时常量池:保存了字节码文件中的常量池内容静态常量池:字节码文件通过编号查表的方式找到常量。
32 0
|
1月前
|
缓存 算法 Java
JVM深入原理(八)(一):垃圾回收
弱引用-作用:JVM中使用WeakReference对象来实现软引用,一般在ThreadLocal中,当进行垃圾回收时,被弱引用对象引用的对象就直接被回收.软引用-作用:JVM中使用SoftReference对象来实现软引用,一般在缓存中使用,当程序内存不足时,被引用的对象就会被回收.强引用-作用:可达性算法描述的根对象引用普通对象的引用,指的就是强引用,只要有这层关系存在,被引用的对象就会不被垃圾回收。引用计数法-缺点:如果两个对象循环引用,而又没有其他的对象来引用它们,这样就造成垃圾堆积。
51 0
|
1月前
|
算法 Java 对象存储
JVM深入原理(八)(二):垃圾回收
Java垃圾回收过程会通过单独的GC线程来完成,但是不管使用哪一种GC算法,都会有部分阶段需要停止所有的用户线程。这个过程被称之为StopTheWorld简称STW,如果STW时间过长则会影响用户的使用。一般来说,堆内存越大,最大STW就越长,想减少最大STW,就会减少吞吐量,不同的GC算法适用于不同的场景。分代回收算法将整个堆中的区域划分为新生代和老年代。--超过新生代大小的大对象会直接晋升到老年代。
47 0
|
3月前
|
存储 算法 Java
G1原理—5.G1垃圾回收过程之Mixed GC
本文介绍了G1的Mixed GC垃圾回收过程,包括并发标记算法详解、三色标记法如何解决错标漏标问题、SATB如何解决错标漏标问题、Mixed GC的过程、选择CollectSet的算法
G1原理—5.G1垃圾回收过程之Mixed GC