When and how are classes garbage collected in Java?(Java 中何时以及如何收集类垃圾?)
问题描述
我在这个主题.但我得到的答案,给了我另一个问题.
有人提到垃圾收集器也可以收集类.这是真的?
如果这是真的,这是怎么回事?
Java 中的类可以在没有任何引用的情况下被垃圾回收.在大多数简单的设置中,这永远不会发生,但在某些情况下可能会发生.
有很多方法可以使类可访问,从而阻止它符合 GC 条件:
- 该类的对象仍然可以访问.
- 表示该类的
Class对象仍然可以访问 - 加载该类的
ClassLoader仍然可以访问 ClassLoader加载的其他类仍然可以访问
当 none 为真时,ClassLoader 及其加载的所有类都符合 GC 条件.
这是一个构建的示例(充满了不良做法!),应该展示这种行为:
在目录(不是包!)x中创建一个字节码文件GCTester.class.它的源代码是:
公共类 GCTester {公共静态最终 GCTester INSTANCE=new GCTester();私人 GCTester() {System.out.println(this + "created");}公共无效finalize(){System.out.println(this + "finalized");}}然后在
x的父目录下创建一个类TestMe:import java.io.File;导入 java.net.URL;导入 java.net.URLClassLoader;导入 java.lang.reflect.Field;公共类TestMe {公共静态 void main(String[] args) 抛出异常 {System.out.println("in main");testGetObject();System.out.println("第二次 gc() 调用 (in main)");System.gc();线程.sleep(1000);System.out.println("main 结束");}公共静态无效 testGetObject() 抛出异常 {System.out.println("创建类加载器");ClassLoader cl = new URLClassLoader(new URL[] {new File("./x").toURI().toURL()});System.out.println("加载类");类<?>clazz = cl.loadClass("GCTester");System.out.println("获取静态字段");字段字段 = clazz.getField("INSTANCE");System.out.println("读取静态值");对象对象 = field.get(null);System.out.println("得到的值:" + object);System.out.println("第一次 gc() 调用");System.gc();线程.sleep(1000);}}运行
<上一页>主要是创建类加载器加载类获取静态字段读取静态值GCTester@1feed786 已创建得到值:GCTester@1feed786第一次 gc() 调用第二次 gc() 调用(在 main 中)GCTester@1feed786 定稿主线结束TestMe将产生这个(或类似的)输出:
在倒数第二行中,我们看到 GCTester 实例已完成,这仅意味着该类(和 ClassLoader)符合垃圾回收条件.
I asked a question about Garbage Collection in Java in this topic. But the answer I got, gave me another question.
Someone mentioned that classes can be collected by the garbage collector too. Is this true?
And if it is true, how does this work?
A class in Java can be garbage-collected when nothing references it. In most simple setups this never happens, but there are situations where it can occur.
There are many ways to make a class reachable and thus prevent it from being eligible for GC:
- objects of that class are still reachable.
- the
Classobject representing the class is still reachable - the
ClassLoaderthat loaded the class is still reachable - other classes loaded by the
ClassLoaderare still reachable
When none of those are true, then the ClassLoader and all classes it loaded are eligible for GC.
Here's a constructed example (full of bad practices!) that should demonstrate the behaviour:
Create a bytecode file GCTester.class in a directory (not package!) x. It's source code is:
public class GCTester {
public static final GCTester INSTANCE=new GCTester();
private GCTester() {
System.out.println(this + " created");
}
public void finalize() {
System.out.println(this + " finalized");
}
}
Then create a class TestMe in the parent directory of x:
import java.io.File;
import java.net.URL;
import java.net.URLClassLoader;
import java.lang.reflect.Field;
public class TestMe {
public static void main(String[] args) throws Exception {
System.out.println("in main");
testGetObject();
System.out.println("Second gc() call (in main)");
System.gc();
Thread.sleep(1000);
System.out.println("End of main");
}
public static void testGetObject() throws Exception {
System.out.println("Creating ClassLoader");
ClassLoader cl = new URLClassLoader(new URL[] {new File("./x").toURI().toURL()});
System.out.println("Loading Class");
Class<?> clazz = cl.loadClass("GCTester");
System.out.println("Getting static field");
Field field = clazz.getField("INSTANCE");
System.out.println("Reading static value");
Object object = field.get(null);
System.out.println("Got value: " + object);
System.out.println("First gc() call");
System.gc();
Thread.sleep(1000);
}
}
Running TestMe will produce this (or similar) output:
in main Creating ClassLoader Loading Class Getting static field Reading static value GCTester@1feed786 created Got value: GCTester@1feed786 First gc() call Second gc() call (in main) GCTester@1feed786 finalized End of main
In the second to last line we see that the GCTester instance is finalized, which can only mean that the class (and ClassLoader) are eligible for garbage collection.
这篇关于Java 中何时以及如何收集类垃圾?的文章就介绍到这了,希望我们推荐的答案对大家有所帮助,也希望大家多多支持编程学习网!
本文标题为:Java 中何时以及如何收集类垃圾?
- Eclipse 插件更新错误日志在哪里? 2022-01-01
- 从 finally 块返回时 Java 的奇怪行为 2022-01-01
- Jersey REST 客户端:发布多部分数据 2022-01-01
- value & 是什么意思?0xff 在 Java 中做什么? 2022-01-01
- 如何使用WebFilter实现授权头检查 2022-01-01
- Safepoint+stats 日志,输出 JDK12 中没有 vmop 操作 2022-01-01
- Java包名称中单词分隔符的约定是什么? 2022-01-01
- Spring Boot连接到使用仲裁器运行的MongoDB副本集 2022-01-01
- 将log4j 1.2配置转换为log4j 2配置 2022-01-01
- C++ 和 Java 进程之间的共享内存 2022-01-01
