欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Java中的守護線程問題

 更新時間:2022年11月09日 16:17:57   作者:Jeffrey_oWang  
這篇文章主要介紹了Java中的守護線程問題,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教

守護線程

在Java中有兩類線程

  • User Thread(用戶線程)
  • Daemon Thread(守護線程)

守護線程的功能非常簡單,在其本身是一個線程的同時,主要是為了給其他的線程提供服務(wù),比如說計時器,清空高速緩存等等操作,守護線程具有和被守護線程一樣的生命周期(這里并不是說守護線程和被守護線程常常是1-1的關(guān)系),當(dāng)被守護線程死亡,守護線程往往也會死亡,當(dāng)虛擬機中只剩下守護線程時,虛擬機就會退出,因為此時也沒有運行程序的必要了

一個比較通俗的解釋:任何一個守護線程都是整個JVM中所有非守護線程的保姆

只要當(dāng)前JVM實例中尚存在任何一個非守護線程沒有結(jié)束,守護線程就全部工作;只有當(dāng)最后一個非守護線程結(jié)束時,守護線程隨著JVM一同結(jié)束工作。

守護線程的作用是為其他線程的運行提供便利服務(wù),守護線程最典型的應(yīng)用就是 GC (垃圾回收器),它就是一個很稱職的守護者。

需要注意的點是:

  • 守護線程的優(yōu)先級比較低
  • 守護線程要注意考慮關(guān)機動作
  • 守護線程應(yīng)該永遠不去訪問固有資源,比如說文件或者數(shù)據(jù)庫,因為它會在任何時候甚至一個操作的中間發(fā)生中斷。

不要給守護線程分擔(dān)讀寫邏輯或者計算邏輯,因為無法確定守護線程是否已經(jīng)完成了工作,但是只要User退出守護線程也會立馬結(jié)束,對于計算機程序來說這樣的程序可能多次運行結(jié)果不一樣,很顯然這對于程序來說是毀滅性的。

操作:

通過thread.setDaemon(true) 將線程轉(zhuǎn)換為守護線程

這個方法必須在thread.start()之前進行調(diào)用

守護線程與用戶線程的區(qū)別

介紹Java守護線程與用戶線程的概念和使用方法,以及相關(guān)注意事項。

1 定義和概述

Java 中的線程分為兩類,分別為daemon 線程(守護線程〉和user 線程(用戶線程)。守護線程又稱Daemon線程,運行在后臺,看不見;用戶線程運行在前臺,看的見。

在JVM啟動時會調(diào)用main 函數(shù), main 函數(shù)所在的線程就是一個用戶線程,其實在JVM內(nèi)部同時-還啟動了好多守護線程, 比如垃圾回收線程。

Daemon線程是一種支持型線程,因為它主要被用作程序中后臺調(diào)度以及支持性工作。這意味著,當(dāng)一個Java虛擬機中不存在非Daemon線程的時候,Java虛擬機將會退出,而不管當(dāng)前是否有守護線程,也就是說守護線程是否結(jié)束并不影響JVM的退出。

實際上,在main線程運行結(jié)束后,JVM會自動啟動一個叫作DestroyJavaVM 的線程,該線程會等待所有用戶線程結(jié)束后終止JVM 進程。

在Tomcat的NIO實現(xiàn)NioEndpoint中會開啟一組接受線程來接受用戶的連接請求,以及一組處理線程負責(zé)具體處理用戶請求,在默認情況下,接受線程和處理線程都是守護線程,這意味著當(dāng)tomcat 收到shutdown 命令后并且沒有其他用戶線程存在的情況下tomcat 進程會馬上消亡,而不會等待處理線程處理完當(dāng)前的請求。

2 使用守護線程

在線程start之前,可以通過調(diào)用thread.setDaemon(true)將線程設(shè)置為Daemon線程。

守護線程有兩種結(jié)束方式:

  • 守護線程也具有自己的run();方法,當(dāng)后臺線程完成自己的run方法后,守護線程結(jié)束。
  • 用戶線程運行結(jié)束,守護線程自動結(jié)束。

3 測試案例

public class Daemon {
//啟動該類,將會構(gòu)造兩條線程,main線程和一條子線程。
    public static void main(String[] args) throws InterruptedException {
        //測試非守護線程
        //可以看到,輸出"main線程結(jié)束"之后,子線程還在繼續(xù)輸出,程序沒有結(jié)束
//        test1();
        //測試守護線程
        //可以看到,輸出"main線程結(jié)束"之后,子線程沒有繼續(xù)輸出,程序結(jié)束
        test2();
    }

    /**
     * 測試非守護線程
     *
     * @throws InterruptedException
     */
    public static void test1() throws InterruptedException {
        Thread thread = new Thread(() -> {
            while (true) {
                try {
                    Thread.currentThread().sleep(500);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
                System.out.println("子線程非守護線程");
            }
        });
        thread.start();
        Thread.currentThread().sleep(1000);
        System.out.println("main線程結(jié)束");
    }

    //測試守護線程
    public static void test2() throws InterruptedException {
        Thread thread = new Thread(() -> {
            while (true) {
                try {
                    Thread.currentThread().sleep(500);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
                System.out.println("子線程守護線程");
            }
        });
        thread.setDaemon(true);
        thread.start();
        Thread.currentThread().sleep(1000);
        System.out.println("main線程結(jié)束");
    }
}

使用jps查看java進程,可以發(fā)現(xiàn)如果子線程是守護線程那么主線程結(jié)束,子線程也結(jié)束了;如果子線程不是守護線程那么主線程結(jié)束,子線程沒有結(jié)束。

4 注意事項

Daemon線程被用作完成支持性工作,但是在Java虛擬機退出時Daemon線程中的finally塊并不一定會執(zhí)行,如下代碼:

public class Daemon {
    public static void main(String[] args) {
        Thread thread = new Thread(new DaemonRunner(), "DaemonRunner");
        thread.setDaemon(true);
        thread.start();
    }

    static class DaemonRunner implements Runnable {
        @Override
        public void run() {
            try {
                SleepUtils.second(10);
            } finally {
                System.out.println("DaemonThread finally run.");
            }
        }
    }
}

運行Daemon程序,可以看到在控制臺上沒有任何輸出。main線程(非Daemon線程)在啟動了線程DaemonRunner之后隨著main方法執(zhí)行完畢而終止,而此時Java虛擬機中已經(jīng)沒有非Daemon線程,虛擬機需要退出。Java虛擬機中的所有Daemon線程都需要立即終止,因此DaemonRunner立即終止,但是DaemonRunner中的finally塊并沒有執(zhí)行。

在構(gòu)建Daemon線程時,不能依靠finally塊中的內(nèi)容來確保執(zhí)行關(guān)閉或清理資源的邏輯。

以上為個人經(jīng)驗,希望能給大家一個參考,也希望大家多多支持腳本之家。

相關(guān)文章

最新評論