您好,登錄后才能下訂單哦!
java中有多少種方式可以實(shí)現(xiàn)多線程?可能很多人對(duì)此沒(méi)有深入了解過(guò),故而小編總結(jié)了這篇文章,通過(guò)這文章的講解,希望你能夠收獲更多知識(shí)。
Java多線程的使用有三種方法:繼承Thread類、實(shí)現(xiàn)Runnable接口和使用Callable和Future創(chuàng)建線程。
一、繼承Thread類
實(shí)現(xiàn)方式很簡(jiǎn)單,只需要?jiǎng)?chuàng)建一個(gè)類去繼承Thread類然后重寫(xiě)run方法,在main方法中調(diào)用該類實(shí)例對(duì)象的start方法即可實(shí)現(xiàn)多線程并發(fā)。代碼:
public class MyThread extends Thread { @Override public void run(){ super.run(); System.out.println("執(zhí)行子線程..."); } }
測(cè)試用例:
public class Test { public static void main(String[] args) { MyThread myThread = new MyThread(); myThread.start(); System.out.println("主線程..."); } }
當(dāng)然,這里的結(jié)果不代表線程的執(zhí)行順序,線程是并發(fā)執(zhí)行的,如果多運(yùn)行幾次,打印順序可能會(huì)不一樣。多線程的運(yùn)行過(guò)程中,CPU是以不確定的方式去執(zhí)行線程的,故運(yùn)行結(jié)果與代碼的執(zhí)行順序或者調(diào)用順序無(wú)關(guān),運(yùn)行結(jié)果也可能不一樣。
這里還有一個(gè)需要注意的點(diǎn)就是main方法中應(yīng)該調(diào)用的是myThread的start方法,而不是run()方法。調(diào)用start()方法是告訴CPU此線程已經(jīng)準(zhǔn)備就緒可以執(zhí)行,進(jìn)而系統(tǒng)有時(shí)間就會(huì)來(lái)執(zhí)行其run()方法。
而直接調(diào)用run()方法,則不是異步執(zhí)行,而是等同于調(diào)用函數(shù)般按順序同步執(zhí)行,這就失去了多線程的意義了。
二、實(shí)現(xiàn)Runnable接口
這種方式的實(shí)現(xiàn)也很簡(jiǎn)單,就是把繼承Thread類改為實(shí)現(xiàn)Runnable接口。代碼如下:
public class MyRunnable implements Runnable { @Override public void run() { System.out.println("執(zhí)行子線程..."); } }
測(cè)試用例:
public class Test { public static void main(String[] args) { Runnable runnable = new MyRunnable(); Thread thread = new Thread(runnable); thread.start(); System.out.println("主線程運(yùn)行結(jié)束!"); } }
運(yùn)行結(jié)果沒(méi)啥好說(shuō)的,這里main中可以看到真正創(chuàng)建新線程還是通過(guò)Thread創(chuàng)建:
Thread thread = new Thread(runnable);
這一步Thread類的作用就是把run()方法包裝成線程執(zhí)行體,然后依然通過(guò)start去告訴系統(tǒng)這個(gè)線程已經(jīng)準(zhǔn)備好了可以安排執(zhí)行。
三、使用Callable和Future創(chuàng)建線程
上面的兩種方式都有這兩個(gè)問(wèn)題:
1、無(wú)法獲取子線程的返回值;
2、run方法不可以拋出異常。
為了解決這兩個(gè)問(wèn)題,我們就需要用到Callable這個(gè)接口了。說(shuō)到接口,上面的Runnable接口實(shí)現(xiàn)類實(shí)例是作為T(mén)hread類的構(gòu)造函數(shù)的參數(shù)傳入的,之后通過(guò)Thread的start執(zhí)行run方法中的內(nèi)容。但是Callable并不是Runnable的子接口,是個(gè)全新的接口,它的實(shí)例不能直接傳入給Thread構(gòu)造,所以需要另一個(gè)接口來(lái)轉(zhuǎn)換一下。
Java5提供了Future接口來(lái)代表Callable接口里call()方法的返回值,并為Future接口提供了一個(gè)實(shí)現(xiàn)類FutureTask,該實(shí)現(xiàn)類的繼承關(guān)系如圖所示:
可以看到,該實(shí)現(xiàn)類不僅實(shí)現(xiàn)了Future接口,還實(shí)現(xiàn)了Runnable接口,所以可以直接傳給Thread構(gòu)造函數(shù)。
而關(guān)于FutureTask的構(gòu)造函數(shù)如下:
所以這里面其實(shí)就是要比上一個(gè)方法再多一個(gè)轉(zhuǎn)換過(guò)程,最終一樣是通過(guò)Thread的start來(lái)創(chuàng)建新線程。有了這個(gè)思路,代碼就很容易理解了:
import java.util.concurrent.Callable; public class MyCallable implements Callable { int i = 0; @Override public Object call() throws Exception { System.out.println(Thread.currentThread().getName()+" i的值:"+ i); return i++; //call方法可以有返回值 } }
測(cè)試:
import java.util.concurrent.Callable; import java.util.concurrent.FutureTask; public class Test { public static void main(String[] args) { Callable callable = new MyCallable(); for (int i = 0; i < 10; i++) { FutureTask task = new FutureTask(callable); new Thread(task,"子線程"+ i).start(); try { //獲取子線程的返回值 System.out.println("子線程返回值:"+task.get() + "\n"); } catch (Exception e) { e.printStackTrace(); } } } }
執(zhí)行結(jié)果(部分):
關(guān)于java中實(shí)現(xiàn)多線程的方式就分享到這里了,希望以上內(nèi)容可以對(duì)大家有一定的幫助,可以學(xué)到更多知識(shí)。如果覺(jué)得文章不錯(cuò),可以把它分享出去讓更多的人看到。
免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。