chinese直男口爆体育生外卖, 99久久er热在这里只有精品99, 又色又爽又黄18禁美女裸身无遮挡, gogogo高清免费观看日本电视,私密按摩师高清版在线,人妻视频毛茸茸,91论坛 兴趣闲谈,欧美 亚洲 精品 8区,国产精品久久久久精品免费

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

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

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

使用stop、suspend方法來中斷線程的壞處在哪?

Android編程精選 ? 來源:CSDN技術(shù)社區(qū) ? 作者:浪舟子 ? 2021-07-26 14:23 ? 次閱讀
加入交流群
微信小助手二維碼

掃碼添加小助手

加入工程師交流群

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

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

我們從中可以看出以下幾點:

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

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

我們舉例來看一下上邊提到的兩點:

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();

}

運行結(jié)果:

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

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

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

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

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

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

1.定義一個變量,由目標線程去不斷的檢查變量的狀態(tài),當變量達到某個狀態(tài)時停止線程。

代碼舉例如下:

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;

}

運行結(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();

}

運行結(jié)果:

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

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

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

二、suspend的落幕suspend方法的作用是掛起某個線程直到調(diào)用resume方法來恢復(fù)該線程,但是調(diào)用了suspend方法后并不會釋放被掛起線程獲取到的鎖,正因如此就給suspend和resume這哥倆貼上了容易引發(fā)死鎖的標簽,當然這也正是導(dǎo)致suspend和resume退出歷史舞臺的罪魁禍首。同樣我們看看java開發(fā)者為suspend的淘汰給出的理由:

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

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

suspend具有天然的死鎖傾向

當某個線程被suspend后,該線程持有的鎖不會被釋放,其他線程也就不能訪問這些資源

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

接下來模擬一下由suspend引起的死鎖場景,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鎖開始執(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開始執(zhí)行”);

try {

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

} catch (InterruptedException e) {

e.printStackTrace();

}

synchronized (o1)

{

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

}

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

}

});

t2.start();

Thread.sleep(1000);

t1.suspend();

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

int i=1/0;

t1.resume();

}

運行結(jié)果:

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

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

作者丨浪舟子

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

編輯:jq

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

    關(guān)注

    5

    文章

    913

    瀏覽量

    43566

原文標題:為什么強烈不推薦使用stop、suspend方法來中斷線程?

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

收藏 人收藏
加入交流群
微信小助手二維碼

掃碼添加小助手

加入工程師交流群

    評論

    相關(guān)推薦
    熱點推薦

    線程的系統(tǒng)

    線程系統(tǒng)的事件響應(yīng)也是在中斷中完成的,但事件的處理是在線程中完成的。在多線程系統(tǒng)中,線程中斷
    發(fā)表于 12-08 07:55

    Linux多線程對比單線程的優(yōu)勢

    ,而單線程則需要通過進程間通信實現(xiàn)?!干舷挛那袚Q開銷小」:線程的上下文切換比進程小,因為它們共享相同的地址空間?!柑岣唔憫?yīng)性」:多線程可以使程序更加響應(yīng)用戶輸入或其他事件,避免阻塞。
    發(fā)表于 12-01 06:11

    切換線程中斷被屏蔽怎么解決?

    退出,已經(jīng)切換到main線程,但是發(fā)現(xiàn)此時中斷是被屏蔽的。 跟代碼發(fā)現(xiàn)rt_schedule最后切換線程時是先調(diào)用rt_hw_context_switch,再
    發(fā)表于 09-29 07:48

    ulog輸出異常-中斷線程打印沖突怎么解決?

    邏輯:can收到數(shù)據(jù)后,can中斷打印test,并把數(shù)據(jù)放到消息隊列,main函數(shù)的消息隊列打印消息內(nèi)容 問題:只要can中斷中有test打印的代碼,整個串口就會卡死; 中斷中無test打印的代碼
    發(fā)表于 09-28 11:44

    rt_sem_take卡住導(dǎo)致線程無法正常運行怎么解決?

    串口接收數(shù)據(jù)后release信號量,接收線程take sem,高頻大數(shù)據(jù)量接受數(shù)據(jù),運行一段時間后接受線程suspend,但是release正常釋放 出現(xiàn)問題問題后查看信息如下: 接受線程
    發(fā)表于 09-23 08:17

    想在rtsmart中使用uart2,是不是只能通過修改設(shè)備樹方法來實現(xiàn)uart2的復(fù)用呀?

    我想在rtsmart中使用uart2,是不是只能通過修改設(shè)備樹方法來實現(xiàn)uart2的復(fù)用呀? 修改設(shè)備樹后如何只編譯設(shè)備樹文件? 編譯生成的文件可以直接替換到廬山派里嗎,具體替換路徑在哪里呀?
    發(fā)表于 06-24 07:04

    第六章 外部中斷

    本章介紹W55MH32的IO口作為外部中斷輸入的使用。先闡述了NVIC(嵌套向量中斷控制器) 的結(jié)構(gòu)、寄存器、優(yōu)先級及相關(guān)函數(shù),再說明EXTI(外部中斷和事件控制器)的功能??驁D及IO口與
    的頭像 發(fā)表于 05-26 16:27 ?1007次閱讀
    第六章 外部<b class='flag-5'>中斷</b>

    STM32U5 IWDG的提前喚醒中斷無法在STOP模式下觸發(fā)怎么解決?

    我使用的是STM32U585系列芯片,開啟了IWDG,并在運行模式下測試提前喚醒中斷正常觸發(fā),但在進入STOP2模式后,提前喚醒中斷無法觸發(fā)導(dǎo)致無法喂狗,看門狗超時后芯片復(fù)位。 手冊中寫道IWDG
    發(fā)表于 04-27 08:05

    STM32U5 IWDG的提前喚醒中斷無法在STOP模式下觸發(fā)怎么解決?

    我使用的是STM32U585系列芯片,開啟了IWDG,并在運行模式下測試提前喚醒中斷正常觸發(fā),但在進入STOP2模式后,提前喚醒中斷無法觸發(fā)導(dǎo)致無法喂狗,看門狗超時后芯片復(fù)位。 手冊中寫道IWDG
    發(fā)表于 04-24 07:50

    使用ad9467-250采集低頻信號,請問有什么方法來提高sfdr嗎?

    你好,我現(xiàn)在在使用ad9467-250采集低頻信號,在測試3Mhz部分時sfdr只有86,采樣頻率是102.4Mhz,請問有什么方法來提高sfdr嗎
    發(fā)表于 04-24 06:05

    STM32U5 IWDG的提前喚醒中斷無法在STOP模式下觸發(fā)怎么解決?

    我使用的是STM32U585系列芯片,開啟了IWDG,并在運行模式下測試提前喚醒中斷正常觸發(fā),但在進入STOP2模式后,提前喚醒中斷無法觸發(fā)導(dǎo)致無法喂狗,看門狗超時后芯片復(fù)位。 手冊中寫道IWDG
    發(fā)表于 04-22 08:14

    Stm32f103 hal庫如果設(shè)置多個外部中斷,只要用螺絲刀碰觸其中一個中斷線,所有的中斷函數(shù)都有可能進入,亂跳,為什么?

    Stm32f103 hal庫如果設(shè)置多個外部中斷,只要用螺絲刀碰觸其中一個中斷線,所有的中斷函數(shù)都有可能進入,亂跳。同一個線路板用標準庫就沒問題。正點原子的開發(fā)版及HAL庫例程也這樣。STM32H562開發(fā)板及HAL庫例程也這樣
    發(fā)表于 03-10 08:07

    PTD08A010W想把1.8V的輸出通過軟件編程的方法來改成3.3V的輸出,怎么實現(xiàn)?

    我現(xiàn)在在用VC707FPGA開發(fā)板,上面用到了PTD08A010W這款電源芯片,這里提供的是12V轉(zhuǎn)1.8V的功能,但現(xiàn)在我想把1.8V的輸出通過軟件編程的方法來改成3.3V的輸出,請問有誰可以提供具體的幫助嗎?下面是實際的電路以及電路原理圖,我要調(diào)的電壓為VADJ_FPGA:
    發(fā)表于 03-03 07:55

    用ADS1258做了一塊采集卡,請問有比較簡單的測試方法來測試我的采集卡的性能和精度嗎?

    您好!我現(xiàn)在用ADS1258做了一塊采集卡,請問有比較簡單的測試方法來測試我的采集卡的性能和精度嗎。 我現(xiàn)在用普通的信號發(fā)生器產(chǎn)生了一個正弦波,1KHz的頻率,用采集卡采集了16*1024個數(shù)
    發(fā)表于 02-10 07:49

    程序中斷的100種寫法

    在編程中,實現(xiàn)程序中斷的方式取決于所使用的編程語言、硬件平臺以及具體的應(yīng)用需求。雖然沒有100種獨立的中斷實現(xiàn)方式,但可以通過不同的方法和技術(shù)實現(xiàn)
    發(fā)表于 01-28 08:25