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

java 中斷線程的幾種方式 interrupt()詳解

 更新時間:2021年11月10日 09:18:58   作者:myseries  
中斷(Interrupt)一個線程意味著在該線程完成任務之前停止其正在進行的一切,有效地中止其當前的操作。這篇文章主要介紹了java 中斷線程的幾種方式 interrupt(),需要的朋友可以參考下

中斷

  中斷(Interrupt)一個線程意味著在該線程完成任務之前停止其正在進行的一切,有效地中止其當前的操作。線程是死亡、還是等待新的任務或是繼續(xù)運行至下一步,就取決于這個程序。雖然初次看來它可能顯得簡單,但是,你必須進行一些預警以實現(xiàn)期望的結果。你最好還是牢記以下的幾點告誡。

  首先,忘掉Thread.stop方法。雖然它確實停止了一個正在運行的線程,然而,這種方法是不安全也是不受提倡的,這意味著,在未來的JAVA版本中,它將不復存在。

如何安全的結束一個正在運行的線程

 Thread類相關的方法

  java.lang.Thread類包含了一些常用的方法,如:start(), stop(), stop(Throwable) ,suspend(), destroy() ,resume()。通過這些方法,我們可以對線程進行方便的操作,但是這些方法中,只有start()方法得到了保留。

  在JDK幫助文檔以及Sun公司的一篇文章《Why are Thread.stop, Thread.suspend and Thread.resume Deprecated?》中都講解了舍棄這些方法的原因。

  簡單來說是因為:使用stop方法雖然可以強行終止正在運行或掛起的線程,但使用stop方法是很危險的,就象突然關閉計算機電源,而不是按正常程序關機一樣,可能會產(chǎn)生不可預料的結果,因此,并不推薦使用stop方法來終止線程。

 那么,我們究竟應該如何停止線程呢?

1、任務中一般都會有循環(huán)結構,只要用一個標記控制住循環(huán),就可以結束任務。

2、如果線程處于了凍結狀態(tài),無法讀取標記,此時可以使用interrupt()方法將線程從凍結狀態(tài)強制恢復到運行狀態(tài)中來,讓線程具備CPU的執(zhí)行資格。

(一):使用退出標志

  當run方法執(zhí)行完后,線程就會退出。但有時run方法是永遠不會結束的,如在服務端程序中使用線程進行監(jiān)聽客戶端請求,或是其他的需要循環(huán)處理的任務。

  在這種情況下,一般是將這些任務放在一個循環(huán)中,如while循環(huán)。如果想使while循環(huán)在某一特定條件下退出,最直接的方法就是設一個boolean類型的標志,并通過設置這個標志為true或false來控制while循環(huán)是否退出。

public class test1 {

    public static volatile boolean exit =false;  //退出標志
    
    public static void main(String[] args) {
        new Thread() {
            public void run() {
                System.out.println("線程啟動了");
                while (!exit) {
                    try {
                        Thread.sleep(1000);
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                }
                System.out.println("線程結束了");
            }
        }.start();
        
        try {
            Thread.sleep(1000 * 5);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        exit = true;//5秒后更改退出標志的值,沒有這段代碼,線程就一直不能停止
    }
}

(二):使用 interrupt 方法

  Thread.interrupt()方法: 作用是中斷線程。將會設置該線程的中斷狀態(tài)位,即設置為true,中斷的結果線程是死亡、還是等待新的任務或是繼續(xù)運行至下一步,就取決于這個程序本身。線程會不時地檢測這個中斷標示位,以判斷線程是否應該被中斷(中斷標示值是否為true)。它并不像stop方法那樣會中斷一個正在運行的線程 

  interrupt()方法只是改變中斷狀態(tài),不會中斷一個正在運行的線程。需要用戶自己去監(jiān)視線程的狀態(tài)為并做處理。支持線程中斷的方法(也就是線程中斷后會拋出interruptedException的方法)就是在監(jiān)視線程的中斷狀態(tài),一旦線程的中斷狀態(tài)被置為“中斷狀態(tài)”,就會拋出中斷異常。這一方法實際完成的是,給受阻塞的線程發(fā)出一個中斷信號,這樣受阻線程檢查到中斷標識,就得以退出阻塞的狀態(tài)。

更確切的說,如果線程被Object.wait, Thread.join和Thread.sleep三種方法之一阻塞,此時調(diào)用該線程的interrupt()方法,那么該線程將拋出一個 InterruptedException中斷異常(該線程必須事先預備好處理此異常),從而提早地終結被阻塞狀態(tài)。如果線程沒有被阻塞,這時調(diào)用 interrupt()將不起作用,直到執(zhí)行到wait(),sleep(),join()時,才馬上會拋出 InterruptedException。

 使用 interrupt() + InterruptedException來中斷線程

  線程處于阻塞狀態(tài),如Thread.sleep、wait、IO阻塞等情況時,調(diào)用interrupt方法后,sleep等方法將會拋出一個InterruptedException:

public static void main(String[] args) {
        Thread thread = new Thread() {
            public void run() {
                System.out.println("線程啟動了");
                try {
                    Thread.sleep(1000 * 100);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
                System.out.println("線程結束了");
            }
        };
        thread.start();

        try {
            Thread.sleep(1000 * 5);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        thread.interrupt();//作用是:在線程阻塞時拋出一個中斷信號,這樣線程就得以退出阻塞的狀態(tài)
    }

 使用 interrupt() + isInterrupted()來中斷線程  

  this.interrupted():測試當前線程是否已經(jīng)中斷(靜態(tài)方法)。如果連續(xù)調(diào)用該方法,則第二次調(diào)用將返回false。在api文檔中說明interrupted()方法具有清除狀態(tài)的功能。執(zhí)行后具有將狀態(tài)標識清除為false的功能。

  this.isInterrupted():測試線程是否已經(jīng)中斷,但是不能清除狀態(tài)標識。

public static void main(String[] args) {
        Thread thread = new Thread() {
            public void run() {
                System.out.println("線程啟動了");
                while (!isInterrupted()) {
                    System.out.println(isInterrupted());//調(diào)用 interrupt 之后為true
                }
                System.out.println("線程結束了");
            }
        };
        thread.start();

        try {
            Thread.sleep(1000);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        thread.interrupt();
        System.out.println("線程是否被中斷:" + thread.isInterrupted());//true
    }

。。。。。。。。。。

來一個綜合的例子:

public class test1 {

    static volatile boolean flag = true;

    public static void main(String[] args) {
        Thread thread = new Thread(new Runnable() {
            @Override
            public void run() {
                System.out.println("開始休眠");
                try {
                    Thread.sleep(100 * 1000);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
                System.out.println("結束休眠,開始死循環(huán)");
                while (flag) {
                }
                System.out.println("------------------子線程結束------------------");
            }
        });
        thread.start();

        Scanner scanner = new Scanner(System.in);
        System.out.println("輸入1拋出一個中斷異常,輸入2修改循環(huán)標志位,輸入3判斷線程是否阻塞,輸入其他結束Scanner\n");
        while (scanner.hasNext()) {
            String text = scanner.next();
            System.out.println("你輸入了:" + text + "\n");
            if ("1".equals(text)) {
                thread.interrupt();
            } else if ("2".equals(text)) {
                flag = false; //如果不設為false,主線程結束后子線程仍在運行
            } else if ("3".equals(text)) {
                System.out.println(thread.isInterrupted());
            } else {
                scanner.close();
                break;
            }
        }
        System.out.println("------------------主線程結束------------------");
    }
}

不能結束的情況

注意下面這種是根本不能結束的情況!

public class Test {
    public static void main(String[] args) {
        Thread thread = new Thread() {
            public void run() {
                System.out.println("線程啟動了");
                while (true) {//對于這種情況,即使線程調(diào)用了intentrupt()方法并且isInterrupted(),但線程還是會繼續(xù)運行,根本停不下來!
                    System.out.println(isInterrupted());//調(diào)用interrupt之后為true
                }
            }
        };
        thread.start();
        thread.interrupt();//注意,此方法不會中斷一個正在運行的線程,它的作用是:在線程受到阻塞時拋出一個中斷信號,這樣線程就得以退出阻塞的狀態(tài)
        while (true) {
            System.out.println("是否isInterrupted:" + thread.isInterrupted());//true
        }
    }
}

關于interrupted()和isInterrupted()方法的注意事項說明

  interrupted()是靜態(tài)方法:內(nèi)部實現(xiàn)是調(diào)用的當前線程的isInterrupted(),并且會重置當前線程的中斷狀態(tài)。

    測試當前線程是否已經(jīng)中斷(靜態(tài)方法)。返回的是上一次的中斷狀態(tài),并且會清除該狀態(tài),所以連續(xù)調(diào)用兩次,第一次返回true,第二次返回false。

  isInterrupted()是實例方法,是調(diào)用該方法的對象所表示的那個線程的isInterrupted(),不會重置當前線程的中斷狀態(tài)

    測試線程當前是否已經(jīng)中斷,但是不能清除狀態(tài)標識。

測試方法驗證:

1:

  第一個紅框中斷的線程是我們自己創(chuàng)建的thread線程,我調(diào)用的interrupted(),由上面源碼可知是判斷當前線程的中斷狀態(tài),當前線程是main線程,我根本沒有中斷過main線程,所以2次調(diào)用均返回“false”

2:

  第一個紅框中斷的線程是當前線程(main線程),我調(diào)用的interrupted(),由上面源碼可知是判斷當前線程的中斷狀態(tài),當前線程是main線程,所以第1次調(diào)用結果返回“true”,因為我確實中斷了main線程

  由源碼可知interrupted()調(diào)用的是isInterrupted(),并會重置中斷狀態(tài),所以第一次調(diào)用之后把中斷狀態(tài)給重置了,從中斷狀態(tài)重置為非中斷狀態(tài),所以第2次調(diào)用的結果返回“false”

3:

  第一個紅框中斷的線程是我們自己創(chuàng)建的thread線程,我調(diào)用的isInterrupted(),由上面源碼可知是判斷執(zhí)行該方法的對象所表示線程的中斷狀態(tài),也就是thread引用所表示的線程的中斷狀態(tài),所以第1次調(diào)用結果返回“true”,

  由源碼可知isInterrupted()不會重置中斷狀態(tài),所以第一次調(diào)用之后沒有把中斷狀態(tài)給重置(從中斷狀態(tài)重置為非中斷狀態(tài)),所以第2次調(diào)用的結果還返回“true”

到此這篇關于java 中斷線程的幾種方式 interrupt()的文章就介紹到這了,更多相關java 中斷線程interrupt()內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!

相關文章

  • 淺談mybatis如何半自動化解耦(推薦)

    淺談mybatis如何半自動化解耦(推薦)

    這篇文章主要介紹了淺談mybatis如何半自動化解耦,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2019-06-06
  • Java中StringUtils工具類的一些用法實例

    Java中StringUtils工具類的一些用法實例

    這篇文章主要介紹了Java中StringUtils工具類的一些用法實例,本文著重講解了isEmpty和isBlank方法的使用,另外也講解了trim、strip等方法的使用實例,需要的朋友可以參考下
    2015-06-06
  • Java案例使用比較排序器comparator實現(xiàn)成績排序

    Java案例使用比較排序器comparator實現(xiàn)成績排序

    這篇文章主要介紹了Java案例使用比較排序器comparator實現(xiàn)成績排序,主要通過案例用TreeSet集合存儲多個學生信息,并遍歷該集合,要按照總分從高到低進行排序,下文介紹需要的朋友可以參考一下
    2022-04-04
  • SpringBoot整合websocket實現(xiàn)即時通信聊天

    SpringBoot整合websocket實現(xiàn)即時通信聊天

    這篇文章主要介紹了SpringBoot整合websocket實現(xiàn)即時通信聊天,實時通信是一個實時通信系統(tǒng),允許兩人或多人使用網(wǎng)絡實時的傳遞文字消息、文件、語音與視頻交流,需要的朋友可以參考下
    2022-05-05
  • Java zookeeper服務的使用詳解

    Java zookeeper服務的使用詳解

    ZooKeeper是一個分布式的,開放源碼的分布式應用程序協(xié)調(diào)服務,是Google的Chubby一個開源的實現(xiàn),是Hadoop和Hbase的重要組件。它是一個為分布式應用提供一致性服務的軟件,提供的功能包括:配置維護、域名服務、分布式同步、組服務等
    2022-08-08
  • Java多線程基本用法總結

    Java多線程基本用法總結

    本篇文章主要總結了Java線程的一些基本的用法。具有一定的參考價值,下面跟著小編一起來看下吧
    2017-01-01
  • 詳解SpringBoot如何創(chuàng)建自定義Starter

    詳解SpringBoot如何創(chuàng)建自定義Starter

    Spring Boot的自動配置機制為開發(fā)人員提供了一種輕松集成和配置各種功能的便捷方式,本文將深入探討在Spring Boot中如何創(chuàng)建自定義Starter,為構建模塊化且易維護的應用提供有力的支持,需要的朋友可以參考下
    2024-02-02
  • Java switch多值匹配操作詳解

    Java switch多值匹配操作詳解

    這篇文章主要介紹了Java switch多值匹配操作詳解,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友可以參考下
    2020-01-01
  • java 利用java反射機制動態(tài)加載類的簡單實現(xiàn)

    java 利用java反射機制動態(tài)加載類的簡單實現(xiàn)

    下面小編就為大家?guī)硪黄猨ava 利用java反射機制動態(tài)加載類的簡單實現(xiàn)。小編覺得挺不錯的,現(xiàn)在就分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2016-09-09
  • Java EE項目中的異常處理總結(一篇不得不看的文章)

    Java EE項目中的異常處理總結(一篇不得不看的文章)

    什么是異常?運行時發(fā)生的可被捕獲和處理的錯誤。這篇文章主要介紹了Java EE項目中的異常處理總結,有需要的可以了解一下。
    2016-11-11

最新評論