本篇文章為大家展示了分析線程和線程安全的5個(gè)步驟分別是什么,內(nèi)容簡(jiǎn)明扼要并且容易理解,絕對(duì)能使你眼前一亮,通過這篇文章的詳細(xì)介紹希望你能有所收獲。
創(chuàng)新互聯(lián)公司是一家專業(yè)提供新巴爾虎右企業(yè)網(wǎng)站建設(shè),專注與成都網(wǎng)站設(shè)計(jì)、成都做網(wǎng)站、外貿(mào)網(wǎng)站建設(shè)、HTML5、小程序制作等業(yè)務(wù)。10年已為新巴爾虎右眾多企業(yè)、政府機(jī)構(gòu)等服務(wù)。創(chuàng)新互聯(lián)專業(yè)網(wǎng)站制作公司優(yōu)惠進(jìn)行中。
什么是線程中斷?
在我們的Java程序中其實(shí)有不止一條執(zhí)行線程,只有當(dāng)所有的線程都運(yùn)行結(jié)束的時(shí)候,這個(gè)Java程序才算運(yùn)行結(jié)束。 官方的話給你描述一下:當(dāng)所有的非守護(hù)線程運(yùn)行結(jié)束時(shí),或者其中一個(gè)線程調(diào)用了System.exit()方法時(shí),這個(gè)Java程序才能運(yùn)行結(jié)束。
線程中斷的應(yīng)用場(chǎng)景
我們先來舉一個(gè)例子,比如我們現(xiàn)在在下載一個(gè)500多M的大片,我們點(diǎn)擊開始下載,那個(gè)這個(gè)時(shí)候就等于開啟了一個(gè)線程去下載我們的文件,然而這個(gè)時(shí)候我們的網(wǎng)速不是很給力,幾十KB的在這跑,作為一個(gè)年輕人我是等不了了,我不下來,那么這個(gè)時(shí)候我們***個(gè)操作就是結(jié)束掉這個(gè)下載文件的操作,其實(shí)更接近程序的來說,這個(gè)時(shí)候我們就需要把這個(gè)線程給中斷了。
我們接下來寫一下這個(gè)下載的代碼,看一下如何中斷一個(gè)線程,這里我已經(jīng)默認(rèn)你們已經(jīng)掌握了如何創(chuàng)建一個(gè)線程了,這段程序我們模擬下載,最開始獲取系統(tǒng)時(shí)間,然后進(jìn)入循環(huán)每次獲取系統(tǒng)時(shí)間,如果時(shí)間超過10秒我們就中斷線程,不在繼續(xù)下載,下載速度時(shí)每秒1M:
public void run() { int number = 0; // 記錄程序開始的時(shí)間 Long start = System.currentTimeMillis(); while (true) { // 每次執(zhí)行一次結(jié)束的時(shí)間 Long end = System.currentTimeMillis(); // 獲取時(shí)間差 Long interval = end - start; // 如果時(shí)間超過了10秒,那么我們就結(jié)束下載 if (interval >= 10000) { // 中斷線程 interrupted(); System.err.println("太慢了,我不下了"); return; } else if (number >= 500) { System.out.println("文件下載完成"); // 中斷線程 interrupted(); return; } number++; System.out.println("已下載" + number + "M"); try { Thread.sleep(2000); } catch (InterruptedException e) { e.printStackTrace(); } } }
中斷線程的方式
Thread類中給我們提供了中斷線程的方法,我們先來看下這個(gè)方法到底是如何讓線程中斷的:
public static boolean interrupted() { return currentThread().isInterrupted(true); }
這個(gè)方法是檢查當(dāng)前線程是否被中斷,中斷返回true,未中斷返回false
private native boolean isInterrupted(boolean ClearInterrupted);
通過查看源碼我們可以發(fā)現(xiàn),中斷線程就是通過調(diào)用檢查線程是否被中斷的方法,把值設(shè)為true。這個(gè)時(shí)候你再去調(diào)用檢查線程是否中斷的方法時(shí)就返回true了。
這里大家需要注意一個(gè)問題:Thread.interrupted()方法只是修改了當(dāng)前線程的狀態(tài)告訴他被中斷了,但是對(duì)于非阻塞中的線程,只是改變了中斷狀態(tài),即 Thread.isInterrupted()返回true,對(duì)于可取消的阻塞狀態(tài)中的線程,例如等待在這些函數(shù)上的線程 ,Thread.sleep(),這個(gè)線程收到中斷信號(hào)之后就會(huì)拋出InterruptedException異常,同時(shí)會(huì)把中斷狀態(tài)設(shè)置為true。
線程睡眠引起InterruptedException異常的原因
其實(shí)這樣說大家也是一知半解,我就寫一個(gè)錯(cuò)誤的示例,大家來看一下,把這個(gè)問題徹底的搞清楚:
public void run() { int number = 0; while (true) { // 檢查線程是否被中斷,中斷就停止下載 if (isInterrupted()) { System.err.println("太慢了,我不下了"); return; } else if (number >= 500) { System.out.println("下載完成"); return; } number++; System.out.println("已下載" + number + "M"); try { Thread.sleep(2000); } catch (InterruptedException e) { e.printStackTrace(); } } }
這是我們的主程序,等待10秒后中斷線程
public static void main(String[] args) throws InterruptedException { Thread thread = new PrimeGenerator(); // 啟動(dòng)線程 thread.start(); // 等待10秒后中斷線程 Thread.sleep(1000); // 中斷線程 thread.interrupt(); }
看起來很通常的一個(gè)程序,但是事實(shí)卻并非你看到的樣子,其實(shí)這段代碼是會(huì)拋出InterruptedException異常的,我們來分析原因。
這里我們先要了解Thread.interrupt()方法不會(huì)中斷一個(gè)正在運(yùn)行的線程,調(diào)用Thread.sleep()方法時(shí),這個(gè)時(shí)候就不再占用CPU,我們來分析下我們這個(gè)程序,我們下載是要等待10秒,每次下載的速度是0.5M/S,也就是當(dāng)我們下載到5M的時(shí)候等待時(shí)間已經(jīng)到了,這個(gè)時(shí)候調(diào)用Thread.interrupt()方法中斷線程,但是run()方法中的睡眠還要接著往下執(zhí)行,它是不會(huì)因?yàn)橹袛喽艞増?zhí)行下面的代碼的,那么這個(gè)時(shí)候當(dāng)它再執(zhí)行Thread.sleep()的時(shí)候就會(huì)拋出InterruptedException異常,因?yàn)楫?dāng)前的線程已經(jīng)被中斷了。
說到這里,你是否已經(jīng)明白產(chǎn)生這個(gè)異常的原因了?另外還有另外的兩個(gè)原因致使線程產(chǎn)生InterruptedException異常的原因,wait()、join()兩個(gè)方法使用不當(dāng)也會(huì)引起線程拋出該異常。
查看線程是否中斷的兩種方式
在Thread類中有一個(gè)方法interrupted()可以用來檢查當(dāng)前線程時(shí)候被中斷,還有isInterrupted()方法可以用來檢查當(dāng)前線程是否被中斷。
中斷線程的方法其實(shí)底層就是將這個(gè)屬性設(shè)置為true,isInterrupted()方法只是返回了這個(gè)屬性值而已。
這兩個(gè)方法有一個(gè)區(qū)別就是isInterrupted()不能改變interrupted()的屬性值,但是interrupted()方法卻能改變interrupted的屬性值,所以在判斷一個(gè)線程時(shí)候被中斷的時(shí)候我們更推薦使用isInterrupted()。
上述內(nèi)容就是分析線程和線程安全的5個(gè)步驟分別是什么,你們學(xué)到知識(shí)或技能了嗎?如果還想學(xué)到更多技能或者豐富自己的知識(shí)儲(chǔ)備,歡迎關(guān)注創(chuàng)新互聯(lián)行業(yè)資訊頻道。