您好,登錄后才能下訂單哦!
這篇文章將為大家詳細(xì)講解有關(guān)如何從Thread和Runnable接口以及Callable方面來創(chuàng)建線程,文章內(nèi)容質(zhì)量較高,因此小編分享給大家做個參考,希望大家閱讀完這篇文章后對相關(guān)知識有一定的了解。
我是線程,我的英文名叫 Thread,別看我現(xiàn)在風(fēng)光無限,好像人盡皆知的樣子,然而我的身世卻悲慘離奇。
接下來我要把我的出生過程演示給你看,這也是我的第一段人生經(jīng)歷。
線程最原始的創(chuàng)建方式,只需要繼承 Thread 類,重寫 run() 方法即可,實(shí)現(xiàn)代碼如下:
// 創(chuàng)建方式 1:繼承 Thread class MyThread extends Thread { @Override public void run() { System.out.println("你好,線程~"); } } // 測試 public class ThreadExample { public static void main(String[] args) { // 創(chuàng)建線程 Thread thread = new MyThread(); // 啟動線程 thread.start(); } }
以上創(chuàng)建線程的方式略顯繁瑣,我們也可以使用匿名對象的方式,在創(chuàng)建 Thread 類的時候就直接重寫 run() 方法,實(shí)現(xiàn)代碼如下:
// 變種 1:匿名方式創(chuàng)建線程 Thread t1 = new Thread() { @Override public void run() { System.out.println("線程變種"); } }; // 啟動線程 t1.start();
Java 語言的設(shè)計(jì)是單繼承,所以當(dāng)繼承了 Thread 之后,就不能再繼承其他類了。
也就是說,如果我一直呆在親生母親(extends Thread)的身邊,那么就得不到好的教育,所以長大之后也注定會普普通通,這可能就是母親把我過繼給遠(yuǎn)房親戚的原因吧。
在 Java 語言中,雖然不能實(shí)現(xiàn)多繼承,但可以實(shí)現(xiàn)多接口,所以我在第二位母親家,過得也算如魚得水。
和繼承 Thread 類差不多,實(shí)現(xiàn) Runnable 接口也是重寫 run() 方法,具體實(shí)現(xiàn)代碼如下:
public class ThreadExample2 { // 創(chuàng)建方式 2:實(shí)現(xiàn) Runnable 接口 static class MyThread implements Runnable { @Override public void run() { System.out.println("你好,線程~"); } } // 代碼測試 public static void main(String[] args) { // 創(chuàng)建 Runnable 子類 MyThread myThread = new MyThread(); // 創(chuàng)建線程 Thread thread = new Thread(myThread); // 啟動線程 thread.start(); } }
以上實(shí)現(xiàn) Runnable 的接口有更簡單的實(shí)現(xiàn)方法,我們可以使用匿名 Runnable 來創(chuàng)建一個線程,如下代碼所示:
// 變種 1:匿名 Runnable 方式 Thread t2 = new Thread(new Runnable() { @Override public void run() { System.out.println("我是線程變種方法~"); } }); // 啟動線程 t2.start();
在 JDK 8 之后,我們可以使用 Lambda 表達(dá)式來操作代碼了,所以對于創(chuàng)建匿名 Runnable 類,我們也有了更簡單的實(shí)現(xiàn)方法,如下代碼所示:
// 變種 2:使用 Lambda 匿名 Runnable 方式 Thread t3 = new Thread(() -> { System.out.println("我是變種 2~"); }); // 啟動線程 t3.start();
注意:以上實(shí)現(xiàn)代碼只支持 JDK 1.8+ 版本。
雖然我的前兩位母親對我都很好,但對于我這樣一個氣宇軒揚(yáng)、骨骼驚奇將來要拯救宇宙和維護(hù)世界和平的少年來說,只在國內(nèi)混未免局限性太大,所以我一直想去大洋彼岸追尋自己的夢想,然而以「前兩位」母親的財(cái)力不足以支撐我這樣做。
然而我的第二個家庭和村里的首富一家是至交,得知我的志向之后,他們一家愿意傾囊相授,舉一家之力幫我去大洋彼岸追尋我的夢想。于是在感激之余,我的第二位母親讓我當(dāng)場認(rèn)下首富一家為我的干爹、干媽。就這樣,我就有了第三位母親了。
前兩種創(chuàng)建方式雖然不錯,但都不能接收線程執(zhí)行之后的返回值,于是在 JDK 1.5 之后就加入了 Callable 和 Futrue,用于接收線程執(zhí)行之后的返回值,具體的實(shí)現(xiàn)代碼如下:
import java.util.Random; import java.util.concurrent.Callable; import java.util.concurrent.ExecutionException; import java.util.concurrent.FutureTask; /** * 線程創(chuàng)建示例 3 */ public class CreateThreadExample3 { // 創(chuàng)建方式 3:實(shí)現(xiàn) Callable 接口 static class MyCallable implements Callable<Integer> { @Override public Integer call() throws Exception { int num = new Random().nextInt(10); System.out.println("生成隨機(jī)數(shù):" + num); return num; } } // 代碼測試 public static void main(String[] args) throws ExecutionException, InterruptedException { // 創(chuàng)建 Callable 子對象 MyCallable callable = new MyCallable(); // 使用 FutureTask 配合 Callable 子對象得到執(zhí)行結(jié)果 FutureTask<Integer> futureTask = new FutureTask<>(callable); // 創(chuàng)建線程 Thread thread = new Thread(futureTask); // 啟動線程 thread.start(); // 得到線程執(zhí)行的結(jié)果 int result = futureTask.get(); System.out.println("主線程中拿到子線程執(zhí)行結(jié)果:" + result); } }
以上代碼的執(zhí)行結(jié)果如下:
從以上結(jié)果可以看出,使用 Callable 配合 FutrueTask 可以正確拿到線程執(zhí)行之后的返回值。而我的故事也在這里結(jié)束了,我最終不負(fù)三位母親所望,雖不能拯救宇宙和維護(hù)世界和平,但卻也能在程序界作出自己的一些貢獻(xiàn),這就是我和我三位母親的故事。
第一人稱“我”(Thread)的視角講了線程創(chuàng)建的三種方式,第一種是繼承 Thread,但因?yàn)?Java 語言不允許多繼承,所以當(dāng)繼承了 Thread 之后就不能繼承其他類了,于是就有了第二種方式實(shí)現(xiàn) Runnable 接口的方式。然而前兩種實(shí)現(xiàn)雖然可以創(chuàng)建線程,但不能接收線程執(zhí)行之后的返回值,于是就有了第三種實(shí)現(xiàn) Callable,通過它我們可以取得線程執(zhí)行之后的返回值。
關(guān)于如何從Thread和Runnable接口以及Callable方面來創(chuàng)建線程就分享到這里了,希望以上內(nèi)容可以對大家有一定的幫助,可以學(xué)到更多知識。如果覺得文章不錯,可以把它分享出去讓更多的人看到。
免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。