Java之對象銷毀和finalize方法的使用
對象的銷毀
在C++中析構(gòu)方法用于釋放資源并且銷毀對象本身。
在Java中,由于GC的存在,我們不需要手動回收內(nèi)存,這大大減少了工作量,也提高了程序的安全性。但是Java也確實(shí)存在一個類似于C++中析構(gòu)的函數(shù)。
finalize方法
重載該方法,用于在類被GC回收的時候執(zhí)行一些操作。
下面是一個類實(shí)現(xiàn)finalize的示例。
Aoo類具有一個int 一個String屬性,重載了toString并且在構(gòu)造其中打印這個對象及其創(chuàng)建時間,在finalize中打印這個對象及調(diào)用時間。
Aoo類
public class Aoo { ? ? ? ?private int id; ? ? ? ?private String name; ? ? ? ?public Aoo(){ ? ? ? ? ? ? ? this(0, null); ? ? ? ?} ? ? ? ?public Aoo(int id, String name){ ? ? ? ? ? ? ? this.id = id; ? ? ? ? ? ? ? this.name = name; ? ? ? ? ? ? ? System.out.println(this.toString() + " now create:" + System.currentTimeMillis()); ? ? ? ?} ? ? ? ? /* ? ? ? ? * 省略get/set/toString ? ? ? ? */ ? ? ? ?protected void finalize() throws Throwable{ ? ? ? ? ? ? ? super.finalize(); ? ? ? ? ? ? ? System.out.println(this.toString() + "now finalize:" + System.currentTimeMillis()); ? ? ? ?} }
首先,一個簡單的測試
main方法
public class FinalizeTest { ? ? ? ?public static void main(String[] args) throws Exception { ? ? ? ? ? ? ? Aoo a = new Aoo(1, "a"); ? ? ? ? ? ? ? a = null; ? ? ? ? ? ? ? System.gc() ? ? ? ? ? ? ? Thread.sleep(2000); ? ? ? ? ? ? ? System.exit(0); ? ? ? ?} }
打印結(jié)果:
id:1 name:a now create:1497547723036
id:1 name:anow finalize:1497547724059
GC對對象的回收
這里手動調(diào)用了GC來清理內(nèi)存,而如果將其注釋掉System.gc();,打印結(jié)果是這樣的:
id:1 name:a now create:1497547846923
也就是說,在沒有特意調(diào)用GC的情況下,finalize方法根本沒有被調(diào)用,也就是說這個對象根本沒有被主動回收。
和想象中的不同,GC的運(yùn)行方式是惰性的,也就是說,在內(nèi)存沒有一處的情況下,GC不會去主動回收對象,為了驗(yàn)證這個想法,我創(chuàng)建了一個線程,用于不斷的消耗內(nèi)存,并且不主動調(diào)用GC。
ThreadA類
public class ThreadA implements Runnable{ ? ? ? ?public void run() { ? ? ? ? ? ? ? List<Integer> list = new ArrayList<Integer>(); ? ? ? ? ? ? ? int i = 0; ? ? ? ? ? ? ? while(true){ ? ? ? ? ? ? ? ? ? ? ?list.add(i); ? ? ? ? ? ? ? ? ? ? ?i++; ? ? ? ? ? ? ? } ? ? ? ?} }
main方法
public class FinalizeTest { ? ? ? ?public static void main(String[] args) throws Exception { ? ? ? ? ? ? ? Aoo a = new Aoo(1, "a"); ? ? ? ? ? ? ? a = null; ? ? ? ? ? ? ? ThreadA ta = new ThreadA(); ? ? ? ? ? ? ? Thread t = new Thread(ta); ? ? ? ? ? ? ? t.start(); ? ? ? ? ? ? ? Thread.sleep(2000); ? ? ? ? ? ? ? System.exit(0); ? ? ? ?} }
打印結(jié)果:
id:1 name:a now create:1497548135268
id:1 name:anow finalize:1497548135386
這一次盡管沒有手動調(diào)用GC,但是finalize方法仍然運(yùn)行了,也就是說,只有在內(nèi)存被消耗、需要GC出面清理內(nèi)存的時候,GC才會運(yùn)行。
這樣的finalize方法確實(shí)不靠譜,連能不能被調(diào)用都不一定,更不用說完成什么特定的操作了,如果需要關(guān)流等回收資源,不如手動調(diào)用一個方法,或者在final塊中統(tǒng)一釋放資源。
在finalize方法中,是否重新給自己指定一個引用來避免被GC回收?
嘗試在finalize方法中重新引用來讓GC無法回收
修改后的Aoo如下
public class Aoo { ? ? ? ?public static Aoo SAVE = null; ? ? ? ?private int id; ? ? ? ?private String name; ? ? ? ?public Aoo(){ ? ? ? ? ? ? ? this(0, null); ? ? ? ?} ? ? ? ?public Aoo(int id, String name){ ? ? ? ? ? ? ? this.id = id; ? ? ? ? ? ? ? this.name = name; ? ? ? ? ? ? ? System.out.println(this.toString() + " now create:" + System.currentTimeMillis()); ? ? ? ?} ? ? ? ? ?/* ? ? ? ? * 省略get/set/toString ? ? ? ? */ ? ? ? ?protected void finalize() throws Throwable{ ? ? ? ? ? ? ? super.finalize(); ? ? ? ? ? ? ? System.out.println(this.toString() + "now finalize:" + System.currentTimeMillis()); ? ? ? ? ? ? ? SAVE = this; ? ? ? ?} }
main方法
public class FinalizeTest { ? ? ? ?public static void main(String[] args) throws Exception { ? ? ? ? ? ? ? Aoo.SAVE = new Aoo(1, "a"); ? ? ? ? ? ? ? Aoo.SAVE = null; ? ? ? ? ? ? ? System.gc(); ? ? ? ? ? ? ? Thread.sleep(500); ? ? ? ? ? ? ? System.out.println(Aoo.SAVE == null? "a is dead" : "a is alive" ); ? ? ? ? ? ? ? ? ? ? ? ? ? ? System.exit(0); ? ? ? ? ? ?? ? ? ? ?} }
打印結(jié)果:
id:1 name:a now create:1497551409195
id:1 name:anow finalize:1497551409201
a is alive
這里看出,Aoo.SAVE對象確實(shí)“復(fù)活了”,不過這樣的操作是有限制的,如果故技重施不會再一次“復(fù)活”該對象。
main方法
public class FinalizeTest { ? ? ? ?public static void main(String[] args) throws Exception { ? ? ? ? ? ? ? Aoo.SAVE = new Aoo(1, "a"); ? ? ? ? ? ? ? Aoo.SAVE = null; ? ? ? ? ? ? ? System.gc(); ? ? ? ? ? ? ? Thread.sleep(500); ? ? ? ? ? ? ? System.out.println(Aoo.SAVE == null? "a is dead" : "a is alive" ); ? ? ? ? ? ? ? Aoo.SAVE = null; ? ? ? ? ? ? ? System.gc(); ? ? ? ? ? ? ? Thread.sleep(500); ? ? ? ? ? ? ? System.out.println(Aoo.SAVE == null? "a is dead" : "a is alive" ); ? ? ? ? ? ? ? System.exit(0); ? ? ? ? ? ? ? ? ?} }
打印結(jié)果:
id:1 name:a now create:1497551587715
id:1 name:anow finalize:1497551587721
a is alive
a is dead
這里注意到,兩次的操作是相同的,而finalize方法只會被系統(tǒng)調(diào)用一次。
如果finalze方法中出現(xiàn)死循環(huán)會發(fā)生什么?
Aoo類
public class Aoo { ? ? ? ?private int id; ? ? ? ?private String name; ? ? ? ?public Aoo(){ ? ? ? ? ? ? ? this(0, null); ? ? ? ?} ? ? ? ?public Aoo(int id, String name){ ? ? ? ? ? ? ? this.id = id; ? ? ? ? ? ? ? this.name = name; ? ? ? ? ? ? ? System.out.println(this.toString() + " now create:" + System.currentTimeMillis()); ? ? ? ?} ? ? ? ?/* ? ? ? ? * 省略get/set/toString ? ? ? ? */ ? ? ? ?protected void finalize() throws Throwable{ ? ? ? ? ? ? ? super.finalize(); ? ? ? ? ? ? ? while(true){ ? ? ? ? ? ? ? ? ? ? ?System.out.println(this.toString() + "now finalize:" + System.currentTimeMillis()); ? ? ? ? ? ? ? ? ? ? ?Thread.sleep(100); ? ? ? ? ? ? ? } ? ? ? ?} }
main方法
public class FinalizeTest { ? ? ? ?public static void main(String[] args) throws Exception { ? ? ? ? ? ? ? Aoo a1 = new Aoo(1 , "a1"); ? ? ? ? ? ? ? Aoo a2 = new Aoo(2 , "a2"); ? ? ? ? ? ? ? a1 = null; ? ? ? ? ? ? ? a2 = null; ? ? ? ? ? ? ? ThreadA ta = new ThreadA(); ? ? ? ? ? ? ? Thread t = new Thread(ta); ? ? ? ? ? ? ? t.start(); ? ? ? ? ? ? ? Thread.sleep(5000); ? ? ? ? ? ? ? System.exit(0); ? ? ? ?} }
打印結(jié)果:
id:1 name:a1 now create:1497552024252
id:2 name:a2 now create:1497552024252
id:1 name:a1now finalize:1497552024373
id:1 name:a1now finalize:1497552024503
id:1 name:a1now finalize:1497552026848
id:1 name:a1now finalize:1497552028960
id:1 name:a1now finalize:1497552032363
結(jié)果是隨機(jī)的,有時候是執(zhí)行的a1的finalize,有的時候執(zhí)行的是a2的。
這個結(jié)果說明了兩點(diǎn):
1.finalze方法在的線程優(yōu)先級很低,時間間隔相當(dāng)?shù)牟淮_定并且明顯大于100毫秒。
2.這個死循環(huán)導(dǎo)致了別的對象的finalize方法無法執(zhí)行。
如果對象的創(chuàng)建出現(xiàn)這種死循環(huán),會不會導(dǎo)致對象無法銷毀進(jìn)而導(dǎo)致內(nèi)存溢出?
我們大量創(chuàng)建Aoo對象,并且等待GC自己回收內(nèi)存。
為了直觀的觀看finalize方法的調(diào)用情況,刪除掉了Aoo對象初始化的時候的打印代碼。
main方法
public class FinalizeTest { ? ? ? ?public static void main(String[] args) throws Exception { ? ? ? ? ? ? ? int i = 1; ? ? ? ? ? ? ? while(true){ ? ? ? ? ? ? ? ? ? ? ?Aoo a = new Aoo(i , "a" + i); ? ? ? ? ? ? ? ? ? ? ?i++; ? ? ? ? ? ? ? } ? ? ? ?} }
讓程序執(zhí)行了約兩分鐘,然后手動終止,查看輸出
1497554225913
id:269614 name:a269614now finalize:1497554226151
id:269614 name:a269614now finalize:1497554227635
id:269614 name:a269614now finalize:1497554227735
id:269614 name:a269614now finalize:1497554227836
id:269614 name:a269614now finalize:1497554229586
id:269614 name:a269614now finalize:1497554229686
id:269614 name:a269614now finalize:1497554229951
id:269614 name:a269614now finalize:1497554230051
id:269614 name:a269614now finalize:1497554230152
id:269614 name:a269614now finalize:1497554233699
id:269614 name:a269614now finalize:1497554233800
id:269614 name:a269614now finalize:1497554233900
id:269614 name:a269614now finalize:1497554234308
id:269614 name:a269614now finalize:1497554234408
id:269614 name:a269614now finalize:1497554234508
id:269614 name:a269614now finalize:1497554235053
id:269614 name:a269614now finalize:1497554235153
id:269614 name:a269614now finalize:1497554235253
id:269614 name:a269614now finalize:1497554235823
id:269614 name:a269614now finalize:1497554235923
id:269614 name:a269614now finalize:1497554236023
id:269614 name:a269614now finalize:1497554240324
id:269614 name:a269614now finalize:1497554240424
id:269614 name:a269614now finalize:1497554240525
id:269614 name:a269614now finalize:1497554241146
id:269614 name:a269614now finalize:1497554241247
id:269614 name:a269614now finalize:1497554241347
id:269614 name:a269614now finalize:1497554241448
id:269614 name:a269614now finalize:1497554242020
id:269614 name:a269614now finalize:1497554242120
id:269614 name:a269614now finalize:1497554242220
id:269614 name:a269614now finalize:1497554242321
id:269614 name:a269614now finalize:1497554242421
id:269614 name:a269614now finalize:1497554242521
id:269614 name:a269614now finalize:1497554248367
id:269614 name:a269614now finalize:1497554248467
id:269614 name:a269614now finalize:1497554248567
id:269614 name:a269614now finalize:1497554248667
id:269614 name:a269614now finalize:1497554249534
id:269614 name:a269614now finalize:1497554249634
id:269614 name:a269614now finalize:1497554249734
id:269614 name:a269614now finalize:1497554249835
id:269614 name:a269614now finalize:1497554255954
id:269614 name:a269614now finalize:1497554256055
id:269614 name:a269614now finalize:1497554256155
id:269614 name:a269614now finalize:1497554256255
id:269614 name:a269614now finalize:1497554256356
id:269614 name:a269614now finalize:1497554257285
id:269614 name:a269614now finalize:1497554257386
id:269614 name:a269614now finalize:1497554257486
id:269614 name:a269614now finalize:1497554257586
id:269614 name:a269614now finalize:1497554257686
id:269614 name:a269614now finalize:1497554268652
id:269614 name:a269614now finalize:1497554268753
id:269614 name:a269614now finalize:1497554268853
id:269614 name:a269614now finalize:1497554268953
id:269614 name:a269614now finalize:1497554269054
id:269614 name:a269614now finalize:1497554269154
id:269614 name:a269614now finalize:1497554277474
id:269614 name:a269614now finalize:1497554292852
id:269614 name:a269614now finalize:1497554301062
可以發(fā)現(xiàn)兩個情況:
1.只有一個對象的finalize方法被執(zhí)行了,也就是說這個死循環(huán)的finalize方法阻止了其他對象執(zhí)行finalize方法
2.程序執(zhí)行很快的一段時間后,finalize方法就開始執(zhí)行,但是隨著內(nèi)存消耗的不斷增加,finalize方法被執(zhí)行的次數(shù)也就越來越少。至于為什么這樣,我不知道= =#
總結(jié)
至此,我嘗試了finalize方法的一些用法和特殊情況??梢钥闯?,GC調(diào)用finalize方法存在巨大的不確定性,確實(shí)很不靠譜,不過通過這個方法,了解了一些關(guān)于GC的知識,也讓我明白,雖然Java語言雖然具有高度的一致性等特點(diǎn)使之很容易上手,但是要做到對Java的精通,路還很遠(yuǎn)呢~~
以上為個人經(jīng)驗(yàn),希望能給大家一個參考,也希望大家多多支持腳本之家。
相關(guān)文章
IDEA中JDK是1.8但Java版本只有21和17的解決辦法
JDK 1.8(Java Development Kit 1.8)是Java平臺的一個版本,它包含了用于開發(fā)和運(yùn)行Java應(yīng)用程序的工具和庫,下面這篇文章主要給大家介紹了關(guān)于IDEA中JDK是1.8但Java版本只有21和17的解決辦法,需要的朋友可以參考下2024-01-01PostgreSQL Docker部署+SpringBoot集成方式
本文介紹了如何在Docker中部署PostgreSQL和pgadmin,并通過SpringBoot集成PostgreSQL,主要步驟包括安裝PostgreSQL和pgadmin,配置防火墻,創(chuàng)建數(shù)據(jù)庫和表,以及在SpringBoot中配置數(shù)據(jù)源和實(shí)體類2024-12-12java HashMap,TreeMap與LinkedHashMap的詳解
這篇文章主要介紹了 java HashMap,TreeMap與LinkedHashMap的詳解的相關(guān)資料,這里提供實(shí)例代碼,幫助大家學(xué)習(xí)理解 這部分的內(nèi)容,需要的朋友可以參考下2016-11-11淺談Java8 的foreach跳出循環(huán)break/return
這篇文章主要介紹了Java8 的foreach跳出循環(huán)break/return,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2021-07-07java實(shí)現(xiàn)基于Tcp的socket聊天程序
這篇文章主要為大家詳細(xì)介紹了java實(shí)現(xiàn)基于Tcp的socket聊天程序,文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下2018-07-07SpringBoot獲取配置文件內(nèi)容的幾種方式總結(jié)
大家都知道SpringBoot獲取配置文件的方法有很多,下面這篇文章主要給大家介紹了關(guān)于SpringBoot獲取配置文件內(nèi)容的幾種方式,文中通過實(shí)例代碼介紹的非常詳細(xì),需要的朋友可以參考下2023-02-02