摘要:垃圾回收机制是守护线程的最佳示例,因为它始终在后台运行。
本文分享自华为云社区《一文带你了解Java 中的垃圾回收机制》,作者:海拥。
Integer i = new Integer(4); // 新的 Integer 对象可通过 'i' 中的引用访问 i = null; // Integer 对象不再可用。
以上所有带有示例的方法都在单独的文章中讨论:如何使对象符合垃圾收集条件
// 演示请求 JVM 运行垃圾收集器的 Java 程序 public class Test { public static void main(String[] args) throws InterruptedException { Test t1 = new Test(); Test t2 = new Test(); // 取消引用变量 t1 = null; // 请求 JVM 来运行垃圾收集器 System.gc(); // 取消引用变量 t2 = null; // 请求 JVM 来运行垃圾收集器 Runtime.getRuntime().gc(); } @Override // 在垃圾回收之前,在对象上调用一次 finalize 方法 protected void finalize() throws Throwable { System.out.println("垃圾收集器调用"); System.out.println("对象垃圾收集:" + this); } }
输出:
垃圾收集器调用 对象垃圾收集:haiyong.Test@7ad74083 垃圾收集器调用 对象垃圾收集:haiyong.Test@7410a1a9
笔记 :
定稿
protected void finalize() throws Throwable
根据我们的要求,我们可以覆盖finalize() 方法来执行我们的清理活动,例如关闭数据库连接。
笔记 :
有关finalize() 方法的示例,请参阅Java 程序的输出第十套之垃圾收集
让我们举一个真实的例子,在那里我们使用垃圾收集器的概念。
假设你去字节跳动实习,他们告诉你写一个程序,计算在公司工作的员工人数(不包括实习生)。要制作这个程序,你必须使用垃圾收集器的概念。
这是您在公司获得的实际任务:-
问: 编写一个程序来创建一个名为 Employee 的类,该类具有以下数据成员。
1.一个ID,用于存储分配给每个员工的唯一ID。
2.员工姓名。
3.员工年龄。
另外,提供以下方法-
现在对垃圾回收机制不了解的初学者可能会这样编写代码:
//计算在公司工作的员工人数的程序 class Employee { private int ID; private String name; private int age; private static int nextId=1; //它是静态的,因为它在所有对象之间保持通用并由所有对象共享 public Employee(String name,int age) { this.name = name; this.age = age; this.ID = nextId++; } public void show() { System.out.println ("Id="+ID+"\nName="+name+"\nAge="+age); } public void showNextId() { System.out.println ("Next employee id will be="+nextId); } } class UseEmployee { public static void main(String []args) { Employee E=new Employee("GFG1",33); Employee F=new Employee("GFG2",45); Employee G=new Employee("GFG3",25); E.show(); F.show(); G.show(); E.showNextId(); F.showNextId(); G.showNextId(); { //这是保留所有实习生的子块。 Employee X=new Employee("GFG4",23); Employee Y=new Employee("GFG5",21); X.show(); Y.show(); X.showNextId(); Y.showNextId(); } //这个大括号之后,X 和 Y 将被移除。因此现在它应该显示 nextId 为 4。 E.showNextId();//这一行的输出应该是 4,但它会给出 6 作为输出。 } }
现在获得正确的输出:
现在垃圾收集器(gc)将看到 2 个空闲的对象。现在递减 nextId,gc(garbage collector) 只会在我们的程序员在我们的类中覆盖它时调用方法 finalize() 。如前所述,我们必须请求 gc(garbage collector),为此,我们必须在关闭子块的大括号之前编写以下 3 个步骤。
现在计算员工人数的正确代码(不包括实习生)
// 计算不包括实习生的员工人数的正确代码 class Employee { private int ID; private String name; private int age; private static int nextId=1; //它是静态的,因为它在所有对象之间保持通用并由所有对象共享 public Employee(String name,int age) { this.name = name; this.age = age; this.ID = nextId++; } public void show() { System.out.println ("Id="+ID+"\nName="+name+"\nAge="+age); } public void showNextId() { System.out.println ("Next employee id will be="+nextId); } protected void finalize() { --nextId; //在这种情况下,gc 会为 2 个对象调用 finalize() 两次。 } } // 它是 Employee 类的右括号 class UseEmployee { public static void main(String []args) { Employee E=new Employee("GFG1",33); Employee F=new Employee("GFG2",45); Employee G=new Employee("GFG3",25); E.show(); F.show(); G.show(); E.showNextId(); F.showNextId(); G.showNextId(); { //这是保留所有实习生的子块。 Employee X=new Employee("GFG4",23); Employee Y=new Employee("GFG5",21); X.show(); Y.show(); X.showNextId(); Y.showNextId(); X = Y = null; System.gc(); System.runFinalization(); } E.showNextId(); } }
点击关注,第一时间了解华为云新鲜技术~