Java并發(fā)編程創(chuàng)建并運行線程的方法對比
一、創(chuàng)建并運行線程的五種方法
第一種:繼承Thread類
這種方式是最基礎的一種方式,學過java的朋友都知道,不做贅述。需要注意的是:覆蓋實現(xiàn)使用的是run方法,運行線程是start方法。
public class FirstWay extends Thread { @Override public void run() { System.out.println("第一種實現(xiàn)線程的方式:繼承Thread類"); } //模擬測試 public static void main(String[] args) { new FirstWay().start(); } }
第二種:實現(xiàn)Runnable接口
第二種實現(xiàn)方式仍然很基礎,繼承Runnable接口,重寫run方法實現(xiàn)線程運行邏輯。需要注意的:運行線程需要套一層new Thread
。
public class SecondWay implements Runnable{ @Override public void run() { System.out.println("第二種實現(xiàn)線程的方式:實現(xiàn)Runnable接口"); } //模擬測試 public static void main(String[] args) { new Thread(new SecondWay()).start(); } }
第三種:實現(xiàn)Callable接口
第三種方式是實現(xiàn)Callable接口,Callable接口與Runable接口都能實現(xiàn)線程。
public class ThirdWay implements Callable<String> { @Override public String call() throws Exception { System.out.println("第三種實現(xiàn)線程的方式:實現(xiàn)Callable接口"); return "Callable接口帶返回值,可以拋出異常"; } //模擬測試 public static void main(String[] args) throws ExecutionException, InterruptedException { FutureTask<String> futureTask = new FutureTask<>(new ThirdWay()); new Thread(futureTask).start(); //阻塞方法,獲取call方法返回值 System.out.println(futureTask.get()); //打?。篊allable接口帶返回值,可以拋出異常 } }
區(qū)別如下:
- Callable接口實現(xiàn)線程方法是call, Runable接口實現(xiàn)線程方法是run
- Callable有返回值, Runable接口不能有返回值
- Callable接口方法call返回值可以設置泛型,如下例子中使用String數(shù)據(jù)類型
- Callable接口方法call方法可以拋出異常,Runable接口run方法不可以拋出異常
- Callable接口方法通過
new Thread(futureTask).start()
運行,F(xiàn)utureTask的get方法可以獲取Callable接口方法call方法的返回值 - 如果Callable接口方法call方法異常,在FutureTask的get方法調用時也會拋出同樣的異常
第四種:線程池 + execute
從JDK5版本開始,java默認提供了線程池的支持,用線程池的方式運行線程可以避免線程的無限擴張導致應用宕機,同時也節(jié)省了線程頻繁創(chuàng)建與銷毀的資源與時間成本。
public class FourthWay implements Runnable{ @Override public void run() { System.out.println(Thread.currentThread().getName() + ":實現(xiàn)線程的方式Runnable接口,但運行方式不一樣,使用線程池"); } public static void main(String[] args) { //創(chuàng)建一個固定大小的線程池 ExecutorService threadPool = Executors.newFixedThreadPool(5); for(int i = 0;i < 10;i++){ threadPool.execute(new FourthWay()); } } }
線程池ExecutorService使用execute方法運行Runnable接口run方法的線程實現(xiàn),execute方法與run方法的共同特點是沒有返回值。
pool-1-thread-5:實現(xiàn)線程的方式Runnable接口,但運行方式不一樣,使用線程池 pool-1-thread-2:實現(xiàn)線程的方式Runnable接口,但運行方式不一樣,使用線程池 pool-1-thread-4:實現(xiàn)線程的方式Runnable接口,但運行方式不一樣,使用線程池 pool-1-thread-4:實現(xiàn)線程的方式Runnable接口,但運行方式不一樣,使用線程池 pool-1-thread-4:實現(xiàn)線程的方式Runnable接口,但運行方式不一樣,使用線程池 pool-1-thread-1:實現(xiàn)線程的方式Runnable接口,但運行方式不一樣,使用線程池 pool-1-thread-4:實現(xiàn)線程的方式Runnable接口,但運行方式不一樣,使用線程池 pool-1-thread-3:實現(xiàn)線程的方式Runnable接口,但運行方式不一樣,使用線程池 pool-1-thread-2:實現(xiàn)線程的方式Runnable接口,但運行方式不一樣,使用線程池 pool-1-thread-5:實現(xiàn)線程的方式Runnable接口,但運行方式不一樣,使用線程池
從上面的結果中可以看出,線程池中包含五個線程。線程運行完成之后并不銷毀,而是還回到線程池,下一次執(zhí)行時從線程池中獲取線程資源再次運行。
第五種:線程池 + submit
下面的例子線程池ExecutorService使用submit方法運行Callable接口call方法的線程實現(xiàn),submit方法與call方法的共同特點是存在返回值。
- Callable接口call方法的返回值可以由泛型定義
- ExecutorService線程池submit方法的返回值是Future
Future的get方法可以獲取call方法的返回值,同時如果call方法拋出異常,F(xiàn)uture的get方法也會拋出異常。
public class FifthWay implements Callable<String> { @Override public String call() throws Exception { return Thread.currentThread().getName() + ":Callable接口帶返回值,可以拋出異常"; } //模擬測試 public static void main(String[] args) throws ExecutionException, InterruptedException { //保存多線程執(zhí)行結果 List<String> retList = new ArrayList<>(); //創(chuàng)建一個固定大小的線程池 ExecutorService threadPool = Executors.newFixedThreadPool(5); for(int i = 0;i < 10;i++){ Future<String> future = threadPool.submit(new FifthWay()); retList.add(future.get()); } //java8 語法,打印retlist retList.forEach(System.out::println); } }
上文代碼中有一個小小的語法糖,retList.forEach(System.out::println);
是java8提供的方法引用
pool-1-thread-1:Callable接口帶返回值,可以拋出異常 pool-1-thread-2:Callable接口帶返回值,可以拋出異常 pool-1-thread-3:Callable接口帶返回值,可以拋出異常 pool-1-thread-4:Callable接口帶返回值,可以拋出異常 pool-1-thread-5:Callable接口帶返回值,可以拋出異常 pool-1-thread-1:Callable接口帶返回值,可以拋出異常 pool-1-thread-2:Callable接口帶返回值,可以拋出異常 pool-1-thread-3:Callable接口帶返回值,可以拋出異常 pool-1-thread-4:Callable接口帶返回值,可以拋出異常 pool-1-thread-5:Callable接口帶返回值,可以拋出異常
總結
本篇文章就到這里了,希望能夠給你帶來幫助,也希望您能夠多多關注腳本之家的更多內容!
相關文章
Java中雙重檢查鎖(double checked locking)的正確實現(xiàn)
雙重檢查鎖(Double-Check Locking),顧名思義,通過兩次檢查,并基于加鎖機制,實現(xiàn)某個功能,下面這篇文章主要給大家介紹了關于Java中雙重檢查鎖(double checked locking)的相關資料,需要的朋友可以參考下2021-09-09解決idea報錯 Connot resolve column 的問題
這篇文章主要介紹了解決idea報錯 Connot resolve column 的問題,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2021-02-02springboot 注冊服務注冊中心(zk)的兩種方式詳解
本文通過一個demo講述一下這兩種注冊方式,使用的是傳統(tǒng)的向zk注冊的方案。對springboot 注冊zk的相關知識感興趣的朋友一起看看吧2018-01-01Spring?Boot實現(xiàn)消息的發(fā)送和接收使用實戰(zhàn)指南
這篇文章主要為大家介紹了Spring?Boot實現(xiàn)消息的發(fā)送和接收使用實戰(zhàn)指南,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2023-06-06