0
  • 聊天消息
  • 系統(tǒng)消息
  • 評(píng)論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會(huì)員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識(shí)你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

使用stop、suspend方法來(lái)中斷線程的壞處在哪?

Android編程精選 ? 來(lái)源:CSDN技術(shù)社區(qū) ? 作者:浪舟子 ? 2021-07-26 14:23 ? 次閱讀

我們知道像stop、suspend這幾種中斷或者阻塞線程的方法在較高java版本中已經(jīng)被標(biāo)記上了@Deprecated過(guò)期標(biāo)簽,那么為什么她們?cè)?jīng)登上了java的歷史舞臺(tái)而又漸漸的推出了舞臺(tái)呢,到底是人性的扭曲還是道德的淪喪呢,亦或是她們不思進(jìn)取被取而代之呢,如果是被取而代之,那么取而代之的又是何方人也,本文我們將一探究竟。

一、stop的落幕首先stop方法的作用是什么呢,用java源碼中的一句注釋來(lái)了解一下:Forces the thread to stop executing.,即強(qiáng)制線程停止執(zhí)行,‘Forces’似乎已經(jīng)透漏出了stop方法的蠻狠無(wú)理。那么我們?cè)倏纯磈ava開(kāi)發(fā)者是怎們解釋stop被淘汰了的。

我們從中可以看出以下幾點(diǎn):

stop這種方法本質(zhì)上是不安全的

使用Thread.stop停止線程會(huì)導(dǎo)致它解鎖所有已鎖定的監(jiān)視器,即直接釋放當(dāng)前線程已經(jīng)獲取到的所有鎖,使得當(dāng)前線程直接進(jìn)入阻塞狀態(tài)

我們舉例來(lái)看一下上邊提到的兩點(diǎn):

public static void main(String[] args) throws InterruptedException {

Object o1=new Object();

Object o2=new Object();

Thread t1=new Thread(()-》{

synchronized (o1)

{

synchronized (o2)

{

try {

System.out.println(“t1獲取到鎖”);

Thread.sleep(5000);

System.out.println(“t1結(jié)束”);

} catch (InterruptedException e) {

e.printStackTrace();

}

}

}

});

t1.start();

Thread.sleep(1000);

Thread t2=new Thread(()-》{

synchronized (o1)

{

synchronized (o2)

{

try {

System.out.println(“t2獲取到鎖”);

Thread.sleep(5000);

System.out.println(“t2結(jié)束”);

} catch (InterruptedException e) {

e.printStackTrace();

}

}

}

});

t2.start();

t1.stop();

}

運(yùn)行結(jié)果:

1fb0958c-e009-11eb-9e57-12bb97331649.png

可以看到,當(dāng)線程t1在獲取到o1和o2兩個(gè)鎖開(kāi)始執(zhí)行,在還沒(méi)有執(zhí)行結(jié)束的時(shí)候,主線程調(diào)用了t1的stop方法中斷了t1的執(zhí)行,釋放了t1線程獲取到的所有鎖,中斷后t2獲取到了o1和o2鎖,開(kāi)始執(zhí)行直到結(jié)束,而t1卻夭折在了sleep的時(shí)候,sleep后的代碼沒(méi)有執(zhí)行。

因此使用stop我們?cè)诓恢谰€程到底運(yùn)行到了什么地方,暴力的中斷了線程,如果sleep后的代碼是資源釋放、重要業(yè)務(wù)邏輯等比較重要的代碼的話,亦或是其他線程依賴t1線程的運(yùn)行結(jié)果,那直接中斷將可能造成很嚴(yán)重的后果。

那么不建議使用stop中斷線程我們應(yīng)該怎么去優(yōu)雅的結(jié)束一個(gè)線程呢,我們可以存java開(kāi)發(fā)者的注釋中窺探到一種解決方案:

1fdbe836-e009-11eb-9e57-12bb97331649.png

可以看到j(luò)ava開(kāi)發(fā)者推薦我們使用以下兩種方法來(lái)優(yōu)雅的停止線程:

1.定義一個(gè)變量,由目標(biāo)線程去不斷的檢查變量的狀態(tài),當(dāng)變量達(dá)到某個(gè)狀態(tài)時(shí)停止線程。

代碼舉例如下:

volatile static boolean flag=false;

public static void main(String[] args) throws InterruptedException {

Object o1=new Object();

Thread t1=new Thread(()-》{

synchronized (o1)

{

try {

System.out.println(“t1獲取到鎖”);

while (!flag)

Thread.sleep(5000);//執(zhí)行業(yè)務(wù)邏輯

System.out.println(“t1結(jié)束”);

} catch (InterruptedException e) {

e.printStackTrace();

}

}

});

t1.start();

Thread.sleep(1000);

Thread t2=new Thread(()-》{

synchronized (o1)

{

try {

System.out.println(“t2獲取到鎖”);

Thread.sleep(5000);//執(zhí)行業(yè)務(wù)邏輯

System.out.println(“t2結(jié)束”);

} catch (InterruptedException e) {

e.printStackTrace();

}

}

});

t2.start();

flag=true;

}

運(yùn)行結(jié)果:

1ffacdc8-e009-11eb-9e57-12bb97331649.png

2.使用interrupt方法中斷線程。

代碼舉例如下:

public static void main(String[] args) throws InterruptedException {

Object o1=new Object();

Thread t1=new Thread(()-》{

synchronized (o1)

{

System.out.println(“t1獲取到鎖”);

while (!Thread.currentThread().isInterrupted()) {

for (int i = 0; i 《 100; i++) {

if(i==50)

System.out.println();

System.out.print(i+“ ”);

}

System.out.println();

}

System.out.println(“t1結(jié)束”);

}

});

t1.start();

Thread t2=new Thread(()-》{

synchronized (o1)

{

try {

System.out.println(“t2獲取到鎖”);

Thread.sleep(5000);//執(zhí)行業(yè)務(wù)邏輯

System.out.println(“t2結(jié)束”);

} catch (InterruptedException e) {

e.printStackTrace();

}

}

});

t2.start();

t1.interrupt();

}

運(yùn)行結(jié)果:

202dbdb4-e009-11eb-9e57-12bb97331649.png

我們用while (!Thread.currentThread().isInterrupted())來(lái)不斷判斷當(dāng)前線程是否被中斷,中斷的話則讓線程自然消亡并釋放鎖??梢钥吹秸{(diào)用interrupt方法后并不會(huì)像stop那樣暴力的中斷線程,會(huì)等到當(dāng)前運(yùn)行的邏輯結(jié)束后再檢查是否中斷,非常的優(yōu)雅。另外,關(guān)注Java知音公眾號(hào),回復(fù)“后端面試”,送你一份面試題寶典!

注:運(yùn)行舉例代碼可能不會(huì)打印出數(shù)字,這是因?yàn)閠1線程運(yùn)行到while(!Thread.currentThread().isInterrupted())時(shí),主線程已經(jīng)調(diào)了interrupt方法,因此多次運(yùn)行可能會(huì)打印出數(shù)字。

二、suspend的落幕suspend方法的作用是掛起某個(gè)線程直到調(diào)用resume方法來(lái)恢復(fù)該線程,但是調(diào)用了suspend方法后并不會(huì)釋放被掛起線程獲取到的鎖,正因如此就給suspend和resume這哥倆貼上了容易引發(fā)死鎖的標(biāo)簽,當(dāng)然這也正是導(dǎo)致suspend和resume退出歷史舞臺(tái)的罪魁禍?zhǔn)住M瑯游覀兛纯磈ava開(kāi)發(fā)者為suspend的淘汰給出的理由:

206731fc-e009-11eb-9e57-12bb97331649.png

從中我們可以得出以下結(jié)論:

suspend具有天然的死鎖傾向

當(dāng)某個(gè)線程被suspend后,該線程持有的鎖不會(huì)被釋放,其他線程也就不能訪問(wèn)這些資源

suspend某個(gè)線程后,如果在resume的過(guò)程中出現(xiàn)異常導(dǎo)致resume方法執(zhí)行失敗,則lock無(wú)法釋放,導(dǎo)致死鎖

接下來(lái)模擬一下由suspend引起的死鎖場(chǎng)景,Talk is cheap,show my code:

public static void main(String[] args) throws InterruptedException {

Object o1=new Object();

Object o2=new Object();

Thread t1=new Thread(()-》{

synchronized (o1)

{

System.out.println(“t1獲取到o1鎖開(kāi)始執(zhí)行”);

try {

Thread.sleep(5000);//模擬執(zhí)行業(yè)務(wù)邏輯

} catch (InterruptedException e) {

e.printStackTrace();

}

System.out.println(“t1執(zhí)行結(jié)束”);

}

});

t1.start();

Thread t2=new Thread(()-》{

synchronized (o2)

{

System.out.println(“t2獲取到o2開(kāi)始執(zhí)行”);

try {

Thread.sleep(2000);//執(zhí)行耗時(shí)業(yè)務(wù)

} catch (InterruptedException e) {

e.printStackTrace();

}

synchronized (o1)

{

System.out.println(“t2獲取到o1鎖開(kāi)始繼續(xù)執(zhí)行”);

}

System.out.println(“t2執(zhí)行結(jié)束”);

}

});

t2.start();

Thread.sleep(1000);

t1.suspend();

//假設(shè)拋出了一個(gè)未知異常

int i=1/0;

t1.resume();

}

運(yùn)行結(jié)果:

20736242-e009-11eb-9e57-12bb97331649.png

可以看到,整個(gè)程序卡的死死的,在調(diào)用resume恢復(fù)t1線程之前拋出了一個(gè)未知異常,導(dǎo)致t1一直掛起進(jìn)而無(wú)法釋放o1鎖,而t2需要獲取到o1鎖后才能繼續(xù)執(zhí)行,但苦苦等待,奈何o1被t1拿捏的死死的,從此整個(gè)程序就陷入了無(wú)盡的等待中----死鎖。

作者丨浪舟子

blog.csdn.net/qq_40400960/article/details/112651249

編輯:jq

聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點(diǎn)僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場(chǎng)。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問(wèn)題,請(qǐng)聯(lián)系本站處理。 舉報(bào)投訴
  • 中斷
    +關(guān)注

    關(guān)注

    5

    文章

    899

    瀏覽量

    41552

原文標(biāo)題:為什么強(qiáng)烈不推薦使用stop、suspend方法來(lái)中斷線程?

文章出處:【微信號(hào):AndroidPush,微信公眾號(hào):Android編程精選】歡迎添加關(guān)注!文章轉(zhuǎn)載請(qǐng)注明出處。

收藏 人收藏

    評(píng)論

    相關(guān)推薦

    socket 多線程編程實(shí)現(xiàn)方法

    在現(xiàn)代網(wǎng)絡(luò)編程中,多線程技術(shù)被廣泛應(yīng)用于提高服務(wù)器的并發(fā)處理能力。Socket編程是網(wǎng)絡(luò)通信的基礎(chǔ),而將多線程技術(shù)應(yīng)用于Socket編程,可以顯著提升服務(wù)器的性能。 多線程編程的基本概念 多
    的頭像 發(fā)表于 11-12 14:16 ?398次閱讀

    斷連塊應(yīng)用于飛控測(cè)試環(huán)境斷線測(cè)試箱

    斷連塊概述Overviewofdisconnectedblocks斷連塊是一種用于電路中斷連接的連接器,廣泛應(yīng)用于電子測(cè)試設(shè)備和通信系統(tǒng)中。它能夠在特定條件下,通過(guò)手動(dòng)插拔精準(zhǔn)地中斷電路連接,以此
    的頭像 發(fā)表于 10-09 08:06 ?269次閱讀
    斷連塊應(yīng)用于飛控測(cè)試環(huán)境<b class='flag-5'>斷線</b>測(cè)試箱

    rtthread是搶占式的rtos,那么線程的timeout參數(shù)具體的作用是什么呢?

    查閱相關(guān)說(shuō)明,rtthread是搶占式的rtos,那么線程的timeout參數(shù)具體的作用是什么呢, 假如線程A,B的優(yōu)先級(jí)分別是1和2,timeout是10ms。當(dāng)線程B 在運(yùn)行中,還未到timeout,此刻掛起的
    發(fā)表于 09-27 08:39

    控制回路斷線可能原因及如何處理

    控制回路斷線是電氣系統(tǒng)中常見(jiàn)的故障之一,它可能導(dǎo)致設(shè)備無(wú)法正常工作,甚至引發(fā)安全事故。本文將分析控制回路斷線的可能原因,并提出相應(yīng)的處理方法。 一、控制回路斷線的可能原因 接線錯(cuò)誤 接
    的頭像 發(fā)表于 08-23 16:36 ?2305次閱讀

    armv8在啟動(dòng)剛開(kāi)始時(shí)調(diào)用了關(guān)中斷,一直到切換第一個(gè)線程,在哪里開(kāi)啟的中斷呢?

    例如當(dāng)前使用的armv8芯片,在啟動(dòng)剛開(kāi)始時(shí)調(diào)用了關(guān)中斷,一直到切換第一個(gè)線程在哪里開(kāi)啟的中斷呢? int rtthread_startup(void
    發(fā)表于 07-04 07:00

    AD8338有沒(méi)有可行的方法來(lái)測(cè)量大增益?

    AD8338是一款VGA芯片,可變?cè)鲆娣秶鸀?10dB至70dB。然而,理論上的70dB增益過(guò)大,導(dǎo)致來(lái)自源的輸入信號(hào)非常小,容易受到干擾,導(dǎo)致70dB增益未得到驗(yàn)證。有沒(méi)有可行的方法來(lái)測(cè)量大增益?
    發(fā)表于 05-21 07:06

    STM32F107的OTG在stop模式下無(wú)法通過(guò)EXTI_18喚醒,沒(méi)有中斷產(chǎn)生是怎么回事?

    作為device,想驗(yàn)證EXTI_18的中斷,配置好EXTI初始化,但只有在enable suspend int后,在插拔USB線纜時(shí)才會(huì)進(jìn)OTG_FS_WKUP_IRQHandler,如果配置設(shè)備
    發(fā)表于 05-06 08:59

    使用FreeRTOS創(chuàng)建的DHCP線程里面的DHCP是在哪里定義的?

    請(qǐng)教下,使用 FreeRTOS 創(chuàng)建的 DHCP 線程里面的 DHCP 是在哪里定義的,貌似沒(méi)有找到 ? FreeRTOS 版本 v9.0.0 具體如下: // 創(chuàng)建 DHCP 線程 void
    發(fā)表于 04-30 07:34

    STM8L151不能同時(shí)初始化同一中斷線的兩個(gè)管腳怎么解決?

    請(qǐng)問(wèn),STM8中斷問(wèn)題,不能同時(shí)初始化同一中斷線的兩個(gè)管腳,比如PC4,PD4,對(duì)應(yīng)中斷線EXTI4,如果同時(shí)初始化PC4/PD4的終端,則中斷不能正常工作。PD4對(duì)應(yīng)按鍵
    發(fā)表于 04-28 07:32

    stm32外部中斷不能喚醒stop休眠模式怎么解決?

    按鍵引腳設(shè)置成外部中斷方式,可以正常喚醒stop模式;但是充電口設(shè)置成外部中斷,卻無(wú)法喚醒stop休眠?。?!配置都是一樣的,按鍵下降沿觸發(fā),充電檢測(cè)是上升沿觸發(fā)。 有可能是什么原因,
    發(fā)表于 04-26 06:11

    如果是使用六步方波的方法來(lái)控制電機(jī),要如何實(shí)現(xiàn)過(guò)電流保護(hù)?

    請(qǐng)問(wèn)一下如果是使用六步方波的方法來(lái)控制電機(jī),要如何實(shí)現(xiàn)過(guò)電流保護(hù)? 以免電流過(guò)大造成板子損壞
    發(fā)表于 04-22 08:10

    stm32f4如何禁止usb suspend或修改進(jìn)入suspend的時(shí)間?

    手冊(cè)說(shuō)是usb設(shè)備在3ms沒(méi)有數(shù)據(jù)就會(huì)進(jìn)入 suspend mode,哪里可以修改時(shí)間,或者把3ms修改成3秒,或者禁止usb 進(jìn)入suspend. DCD_HandleUSBSuspend_ISR USBD_Suspend D
    發(fā)表于 04-02 08:06

    看一下通過(guò)采用HPC方法來(lái)解決汽車行業(yè)工程挑戰(zhàn)的兩個(gè)具體實(shí)例

    大家可能都知道,將Ansys解決方案與高性能計(jì)算(HPC)相結(jié)合,可帶來(lái)巨大的投資回報(bào)(ROI)。接下來(lái),讓我們來(lái)看一下通過(guò)采用HPC方法來(lái)解決汽車行業(yè)工程挑戰(zhàn)的兩個(gè)具體實(shí)例。
    的頭像 發(fā)表于 03-12 14:29 ?593次閱讀

    加濕器的作用及好處與壞處 加濕器的正確使用方法

    疾病的發(fā)生。然而,過(guò)度使用加濕器或錯(cuò)誤使用也可能會(huì)導(dǎo)致空氣中的細(xì)菌和霉菌增加,對(duì)健康產(chǎn)生負(fù)面影響。以下是關(guān)于加濕器的作用及好處與壞處以及正確使用方法。 加濕器的作用及好處: 改善干燥環(huán)境:加濕器能夠在干燥環(huán)境
    的頭像 發(fā)表于 01-25 09:52 ?2062次閱讀

    火線斷線和零線斷線的危害

    在我們?nèi)嗨木€的供電回路里,三根火線和一根零線都是不能缺少的,任何一根斷線都會(huì)給我們的用電設(shè)備造成嚴(yán)重的危害,那么火線斷線和零線斷線的造成后果是一樣的嗎?
    的頭像 發(fā)表于 01-17 10:02 ?3292次閱讀
    火線<b class='flag-5'>斷線</b>和零線<b class='flag-5'>斷線</b>的危害