当前位置: 首页 > 后端技术 > Java

面试必答题:Java垃圾回收

时间:2023-04-01 23:29:16 Java

总结:垃圾回收是守护线程最好的例子,因为它总是在后台运行。本文分享自华为云社区《一文带你了解Java 中的垃圾回收机制》,作者:海勇。简介?在C/C++中,程序员负责对象的创建和销毁。程序员通常会忽略未使用对象的销毁。由于这种疏忽,在某些时候,可能没有足够的可用内存来创建新对象,整个程序将中止,从而导致OutOfMemoryErrors。?但在Java中,程序员不需要关心所有未使用的对象。垃圾回收机制会自动销毁这些对象。?垃圾收集机制是守护线程的最好例子,因为它总是在后台运行。?垃圾收集机制的主要目标是通过销毁无法访问的对象来释放堆内存。重要术语:无法访问的对象:如果一个对象不包含对它的任何引用,则称该对象是无法访问的。另请注意,属于孤岛的对象也是不可访问的。Integeri=newInteger(4);//新的Integer对象可以通过'i'中的引用访问i=null;//Integer对象不再可用。垃圾收集的资格:如果对象无法访问,则称该对象有资格进行GC(垃圾收集)。在上图中,在i=null之后;堆区中的整数对象4符合垃圾回收条件。使对象符合GC条件的方法?即使程序员不负责销毁未使用的对象,如果不再需要,强烈建议使对象不可访问(从而符合GC条件)。?通常有四种不同的方法可以使对象符合垃圾收集的条件。取消引用变量重新分配引用变量对象在方法内部创建的隔离岛以上所有方法和示例都在单独的文章中讨论:如何使对象符合垃圾收集条件让JVM运行垃圾收集器的方法一旦我们创建了对象符合垃圾收集条件,垃圾收集器可能不会立即销毁它。只有当JVM运行垃圾收集器程序时,对象才会被销毁。但是JVM什么时候运行GarbageCollector,我们无法预测。?我们还可以请求JVM运行垃圾收集器。有两种方法可以做到这一点:使用System.gc()方法:System类包含一个静态方法gc()来请求JVM运行垃圾收集器。使用Runtime.getRuntime().gc()方法:运行时类允许应用程序与运行该应用程序的JVM进行交互。因此,通过使用它的gc()方法,我们可以请求JVM运行垃圾收集器。//演示请求JVM运行垃圾收集器的Java程序publicclassTest{publicstaticvoidmain(String[]args)throwsInterruptedException{Testt1=newTest();测试t2=新测试();//解引用变量t1=null;//请求JVM运行垃圾收集器System.gc();//解引用变量t2=null;//请求JVM运行垃圾收集器Runtime.getRuntime().gc();}@Override//在垃圾收集之前,在对象上调用一次finalize方法protectedvoidfinalize()throwsThrowable{System.out.println("Garbagecollectorcall");System.out.println("对象垃圾回收:"+this);}}输出:垃圾收集器调用对象垃圾收集:haiyong.Test@7ad74083垃圾收集器调用对象垃圾收集:haiyong.Test@7410a1a9注意:?无法保证以上两种方法中的任何一种都会运行垃圾收集器。?调用System.gc()等同于调用:Runtime.getRuntime().gc()Finalize?就在对象被销毁之前,垃圾收集器调用对象的finalize()方法来执行清理活动。一旦finalize()方法完成,对象就会被垃圾收集器销毁。?finalize()方法存在于具有以下原型的Object类中。protectedvoidfinalize()throwsThrowable根据我们的要求,我们可以覆盖finalize()方法来执行我们的清理活动,例如关闭数据库连接。注意:finalize()方法由垃圾收集器而不是JVM调用。虽然垃圾收集器是JVM的模块之一。对象类的finalize()方法有自由实现,因此建议重写finalize()方法来处理系统资源或执行其他清理工作。对于任何给定的对象,finalize()方法永远不会被调用超过一次。如果finalize()方法抛出未捕获的异常,则忽略该异常并中止对象的终结。有关finalize()方法的示例,请参阅Java程序的输出。设置10.垃圾收集让我们举一个使用垃圾收集器概念的真实示例。假设你去字节跳动实习,他们让你写一个程序来统计在公司工作的员工人数(不包括实习生)。要编写此程序,您必须使用垃圾收集器的概念。这是您在公司中获得的实际任务:-问:编写一个程序来创建一个名为Employee的类,该类具有以下数据成员。1.用于存储分配给每个员工的唯一ID的ID。2.员工姓名。3.雇员的年龄。此外,还提供了以下方法-用于初始化姓名和年龄的参数化构造函数。ID应在此构造函数中初始化。方法show()显示ID、姓名和年龄。显示下一位员工ID的showNextId()方法。不了解垃圾回收机制的初学者可能会这样写代码:私有字符串名称;私人年龄;私人静态intnextId=1;//它是静态的,因为它在所有对象之间保持通用和共享publicEmployee(Stringname,intage){this.name=name;这个。年龄=年龄;这个.ID=nextId++;}publicvoidshow(){System.out.println("Id="+ID+"\nName="+name+"\nAge="+age);}publicvoidshowNextId(){System.out.println("下一个员工ID为="+nextId);}}classUseEmployee{publicstaticvoidmain(String[]args){EmployeeE=newEmployee("GFG1",33);员工F=新员工("GFG2",45);员工G=新员工("GFG3",25);E.show();F.show();G.show();E.showNextId();F.showNextId();//这是保留所有实习生的子块。员工X=新员工("GFG4",23);员工Y=新员工("GFG5",21);X.show();Y.show();X.showNextId();Y.showNextId();}//在这个大括号之后,X和Y将被移除。所以现在它应该将nextId显示为4。E.showNextId();//这行的输出应该是4,但它给出了6作为输出。}}现在得到正确的输出:现在垃圾收集器(gc)将看到2个空闲对象。现在递减nextId,如果我们的程序员在我们的类中覆盖它,gc(垃圾收集器)只会调用方法finalize()。如前所述,我们必须请求gc(垃圾收集器),为此我们必须在关闭子块的大括号之前编写以下3个步骤。将引用设置为null(即X=Y=null;)调用,System.gc();调用,System.runFinalization();nowthecorrectcodetocalculatethenumberofemployees(exclusioninterns)//计算不包括interns私有字符串名称;私人年龄;私人静态intnextId=1;//它是静态的,因为它在所有对象之间保持通用并由所有对象共享publicEmployee(Stringname,intage){this.name=name;这个。年龄=年龄;这个.ID=nextId++;}publicvoidshow(){System.out.println("Id="+ID+"\nName="+name+"\nAge="+age);}publicvoidshowNextId(){System.out.println("下一个员工ID为="+nextId);}protectedvoidfinalize(){--nextId;//在这种情况下,gc将为2个对象调用两次finalize()。}}//是Employee类的右括号classUseEmployee{publicstaticvoidmain(String[]args){EmployeeE=newEmployee("GFG1",33);员工F=新员工("GFG2",45);员工G=新员工("GFG3",25);E.show();F.show();G.show();E.showNextId();F.showNextId();G.showNextId();{//这是包含所有实习生的子块。员工X=新员工("GFG4",23);员工Y=新员工("GFG5",21);X.show();Y.show();X.showNextId();Y.showNextId();X=Y=空;系统.gc();System.runFinalization();}E.showNextId();}}点击关注,第一时间了解华为云的新鲜技术~