激情久久久_欧美视频区_成人av免费_不卡视频一二三区_欧美精品在欧美一区二区少妇_欧美一区二区三区的

服務器之家:專注于服務器技術及軟件下載分享
分類導航

PHP教程|ASP.NET教程|JAVA教程|ASP教程|

服務器之家 - 編程語言 - JAVA教程 - 詳解java中finalize的實現與相應的執行過程

詳解java中finalize的實現與相應的執行過程

2020-06-14 11:47i flym JAVA教程

在常規的java書籍中,即會描述 object的finalize方法是用于一些特殊的對象在回收之前再做一些掃尾的工作,但是并沒有說明此是如何實現的.本篇從java的角度(不涉及jvm以及c++),有需要的朋友們可以參考借鑒。

FinalReference引用

此類是一個package類型,表示它并不是公開的一部分,繼承自Reference, 即表示也是一種特定的引用類型,因此每個包裝在其中的對象在被回收之前,自己都會放到指定的referqyebceQueue當中.

這個引用對象專門為帶finalize方法的類服務,可以理解為每一個有相應的方法的對象,其都會封裝為一種finalRefernece對象.

因為finalize方法是object定義的,其默認實現為空.那么如果重寫了此方法,那么方法體肯定不為空.即可以通過這一種區別來.只要finalize方法實現不為空的類,此產生的對象都需要被注冊到finalRefernece中.

這一步可以通過在newInstance的時候,即調用object默認構造方法的時候,就可以進行相應的注冊了.

Finalizer#register方法

主要調用了此方法,就會產生相應的finalizer對象,而finalizer對象是繼承于finalReference的.此方法聲明如下:

?
1
2
3
4
/* Invoked by VM */
static void register(Object finalizee) {
 new Finalizer(finalizee);
}

從上面注釋可以看出,此方法會被jvm在特定時期調用.
然后切換到Finalizer的構造方法,如下所示:

?
1
2
3
4
private Finalizer(Object finalizee) {
 super(finalizee, queue);
 add();
}

可以看出,相應的引用對象會通過queue進行回調.add的作用在于將所有還未進行finalize方法的對象存起來,在最后System.shutdown時調用.通過Runtime#runFinalizersOnExit進行設置.

ReferenceQueue

此引用隊列會在相應reference對象的內部對象被回收之前放到此隊列中(詳細說明在另一篇關于reference中再說明.),因為只需要從此隊列中拿到相應的對象,那么此對象就肯定是準備被回收的.

那么在回收之前調用相應的finalize方法即可.

FinalizerThread線程

此線程即是從queue里面,不停的獲取數據,然后調用相應的finalize方法.相應的代碼如下所示:

?
1
2
3
4
5
6
7
8
for (;;) {
 try {
  Finalizer f = (Finalizer)queue.remove();
  f.runFinalizer(jla);
 } catch (InterruptedException x) {
  // ignore and continue
 }
}

而相應的runFinalizer如下所示:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
synchronized (this) {
 if (hasBeenFinalized()) return;
 remove();
}
try {
 Object finalizee = this.get();
 if (finalizee != null && !(finalizee instanceof java.lang.Enum)) {
  jla.invokeFinalize(finalizee);
 
  /* Clear stack slot containing this variable, to decrease
   the chances of false retention with a conservative GC */
  finalizee = null;
 }
} catch (Throwable x) { }
 
super.clear();

在上面的邏輯當中,首先調用remove將其從未finalize中移除.這個方法是保證每個對象的finalize最多只會被調用一次,即當前這次調用完了.它就會被記相應的狀態,即hasBeenFinalized返回為true(其實就是把里面的next指針指向自己.即自己從未finalize中移除,同時也不需要再次調用finalize了).

接下來就是調用相應的finalize方法,上面的jla.invokeFinalize其實就是調用相應對象的finalize方法. 在這個處理中,首先通過get獲取原始對象.在整個jvm處理中,針對finalizeReference在回收之前默認是不將引用設置為null.因為這里,總是能夠獲取相應的引用對象.

處理完之后,最后調用相應的clear,清除相應的引用.這樣達到最終引用沒有其它對象可引用的效果.

在上面的處理當中,并沒有限定調用finalize的時間.因此,一旦如果某個對象的finalize調用慢,就會影響到整個回收鏈的執行,這下就會產生相應的OOM異常了.因此,除非特殊情況,就不要重寫finalize,相應的場景都應該有其它方法可以處理.比如guava中的FinalizableReference.

finalizer啟動線程

在上面的線程,在相應的進程啟動過程中就會被啟動.可以理解為,對象通過調用register(object)觸發finalizer類的初始化.然后,在靜態初始化塊當中,就會啟動相應的回收線程.相應的初始化代碼如下所示:

?
1
2
3
4
5
6
7
8
9
10
static {
 ThreadGroup tg = Thread.currentThread().getThreadGroup();
 for (ThreadGroup tgn = tg;
   tgn != null;
   tg = tgn, tgn = tg.getParent());
 Thread finalizer = new FinalizerThread(tg);
 finalizer.setPriority(Thread.MAX_PRIORITY - 2);
 finalizer.setDaemon(true);
 finalizer.start();
}

上面的static是靜態初始化塊,即只要類Finalizer被使用,即會觸發相應的調用.這里使用的線程組是系統線程組,優先級也還算高,被配置為后臺線程.

在使用jstack打印線程時,出現的如圖下所示的線程,即是由這里來啟動的.如下圖所示

詳解java中finalize的實現與相應的執行過程

總結

整個Finalizer即是通過finalReference,由JVM和相應的java類相互配合來協同工作.并不是全部由jvm實現,因此可以認為其也并不是太底層的東西,而是為了實現相應的語義.一切都是正常的java來完成,由jvm配合.了解到整個過程,也是對java本身的運行機制有所了解.

延伸 · 閱讀

精彩推薦
主站蜘蛛池模板: 一区二区三区在线视频观看58 | 国产精品国产三级国产aⅴ无密码 | 久久艹逼| 欧美日韩一区,二区,三区,久久精品 | 九七在线视频 | 成人国产在线看 | 法国性xxx精品hd专区 | 亚洲精品7777xxxx青睐 | 97精品视频在线观看 | 亚洲小视频在线播放 | 成人综合免费视频 | 久久精品亚洲一区二区 | 午夜激情视频网站 | 亚洲网站免费看 | 深夜视频福利 | 蜜桃一本色道久久综合亚洲精品冫 | 万圣街在线观看免费完整版 | 免费一级特黄欧美大片勹久久网 | 亚洲精品a在线观看 | 性欧美日本 | 日本黄色大片免费 | 自拍偷拍亚洲图片 | 中文日韩字幕 | 成码无人av片在线观看网站 | 国产一级一国产一级毛片 | chengrenzaixian | 国产日产久久久久久 | 中日韩免费视频 | 成人毛片在线免费观看 | 亚洲成人激情在线 | 色综合视频| 国产午夜亚洲精品理论片大丰影院 | 国产精品久久久久久久hd | 欧美成人免费香蕉 | 国产亚洲黑人性受xxxx精品 | 久久免费观看一级毛片 | 免费亚洲视频在线观看 | 91系列在线观看 | 不卡国产一区二区三区四区 | 在线天堂中文在线资源网 | 九色成人在线 |