今天就跟大家聊聊有關(guān)使用finalize方法對垃圾回收有哪些影響,可能很多人都不太了解,為了讓大家更加了解,小編給大家總結(jié)了以下內(nèi)容,希望大家根據(jù)這篇文章可以有所收獲。
創(chuàng)新互聯(lián)公司堅持“要么做到,要么別承諾”的工作理念,服務(wù)領(lǐng)域包括:網(wǎng)站設(shè)計制作、成都做網(wǎng)站、企業(yè)官網(wǎng)、英文網(wǎng)站、手機端網(wǎng)站、網(wǎng)站推廣等服務(wù),滿足客戶于互聯(lián)網(wǎng)時代的新津縣網(wǎng)站設(shè)計、移動媒體設(shè)計的需求,幫助企業(yè)找到有效的互聯(lián)網(wǎng)解決方案。努力成為您成熟可靠的網(wǎng)絡(luò)建設(shè)合作伙伴!
java提供了一個finalize方法,可以幫助我們進(jìn)行資源釋放,類似于C++中的析構(gòu)函數(shù)。但是目前普遍的認(rèn)識是不要使用,為什么呢?就是因為對java虛擬機的垃圾回收有影響。
我們都知道一個對象如果沒有了任何引用,java虛擬機就認(rèn)為這個對象沒什么用了,就會對其進(jìn)行垃圾回收,但是如果這個對象包含了finalize函數(shù),性質(zhì)就不一樣了。怎么不一樣了呢?
java虛擬機在進(jìn)行垃圾回收的時候,一看到這個對象類含有finalize函數(shù),就把這個函數(shù)交給FinalizerThread處理,而包含了這個finalize的對象就會被添加到FinalizerThread的執(zhí)行隊列,并使用一個鏈表,把這些包含了finalize的對象串起來。
他的影響在于只要finalize沒有執(zhí)行,那么這些對象就會一直存在堆區(qū),不過這里只是4個包含了finalize的對象,影響不是那么大,如果有一萬個或者是十萬個呢?這就影響大了。
finalize的原理其實很簡單,在這里簡要的梳理一下:
(1)對象在初始化的過程中會判斷是否重寫了finalize,方法是判斷兩個字段標(biāo)志has_finalizer_flag和RegisterFinalizersAtInit。
(2)如果重寫了finalize,那就把當(dāng)前對象注冊到FinalizerThread的ReferenceQueue隊列中。注冊之后的對象就叫做Finalizer。方法是調(diào)用register_finalizer函數(shù)。此時java虛擬機一看當(dāng)前有這個對象的引用,于是就不進(jìn)行垃圾回收了。
(3)對象開始被調(diào)用,F(xiàn)inalizerThread線程負(fù)責(zé)從ReferenceQueue隊列中獲取Finalizer對象。開始執(zhí)行finalize方法,在執(zhí)行之前,這個對象一直在堆中。
(4)對象執(zhí)行完畢之后,將這個Finalizer對象從隊列中移除,java虛擬機一看對象沒有引用了,就進(jìn)行垃圾回收了。
這就是整個過程。不過在這里我們主要看的是finalize方法對垃圾回收的影響,其實就是在第三步,也就是這個對象含有finalize,進(jìn)入了隊列但一直沒有被調(diào)用的這段時間,會一直占用內(nèi)存。
我們使用一個案例來分析一波:
我們創(chuàng)建一個類
1public class TestFinalizer {
2 public static class Fdd {
3 //分配1M
4 private byte[] content = new byte[1024*1024];
5 @Override
6 protected void finalize() {
7 System.out.println("finalize被執(zhí)行");
8 }
9 }
10 public static void main(String[] args) {
11 for (int i = 0; i < 1000; i++) {
12 Fdd fdd = new Fdd();
13 }
14 }
15}
現(xiàn)在創(chuàng)建了類,我們設(shè)置一下參數(shù)。
1# 最大堆內(nèi)存
2-Xmx5m
3# 最小堆內(nèi)存
4-Xms5m
5# 堆內(nèi)存溢出錯誤打印
6-XX:+HeapDumpOnOutOfMemoryError
7# 把堆相關(guān)信息保存在下列路徑
8-XX:HeapDumpPath=F:/a.dump
在main方法中,創(chuàng)建了1000個Fdd對象,如果不執(zhí)行finalize方法,那么因為沒有調(diào)用所以會進(jìn)行垃圾回收,此時不斷我們創(chuàng)建多少個,都不會出現(xiàn)任何問題。但是如果存在finalize方法,就不一樣了。
1java.lang.OutOfMemoryError: Java heap space
2Dumping heap to F:/a.dump ...
3finalize被執(zhí)行
4finalize被執(zhí)行
5finalize被執(zhí)行
6finalize被執(zhí)行
7finalize被執(zhí)行
8finalize被執(zhí)行
9finalize被執(zhí)行
10Unable to create F:/a.dump: File exists
11Exception in thread "main" java.lang.OutOfMemoryError: Java heap space
12 at com.fdd.chapter2.TestFinalizer$Fdd.(TestFinalizer.java:6)
13 at com.fdd.chapter2.TestFinalizer.main(TestFinalizer.java:14)
我們看到每個對象都會執(zhí)行finalize,在執(zhí)行之前的這段時間一直會在堆區(qū),執(zhí)行完了就會被清理,所以你看到這里執(zhí)行了不少于5次的finalize方法。但是對象一旦超出了我們設(shè)置的5M,就會出現(xiàn)內(nèi)存溢出。一句話總結(jié)就是出現(xiàn)了對象堆積?,F(xiàn)在使用MAT工具來分析一下。
Mat工具是一個插件,也可以自己下載一個。下載完成之后打開我們剛剛生成的a.dump即可。
下面這張圖就是分析的結(jié)果:
a這塊的內(nèi)容就是Finalizer,也就是我們的Fdd對象,b包含的比較多,亂七八糟的剩余信息。當(dāng)然你也可以查看一些其他的信息。都在MAT工具上。還有一些正在執(zhí)行的finalizer和準(zhǔn)備執(zhí)行的。
看完上述內(nèi)容,你們對使用finalize方法對垃圾回收有哪些影響有進(jìn)一步的了解嗎?如果還想了解更多知識或者相關(guān)內(nèi)容,請關(guān)注創(chuàng)新互聯(lián)行業(yè)資訊頻道,感謝大家的支持。