Java面試題:Java中怎麼樣實現多線程

来源:https://www.cnblogs.com/shsxt/archive/2019/02/26/10435179.html
-Advertisement-
Play Games

方法一:繼承 Thread 類,覆蓋方法 run(),我們在創建的 Thread 類的子類中重寫 run() ,加入線程所要執行的代碼即可。 下麵是一個例子: 這種方法簡單明瞭,符合大家的習慣,但是,它也有一個很大的缺點,那就是如果我們的類已經從一個類繼承(如小程式必須繼承自 Applet 類),則 ...


 

方法一:繼承 Thread 類,覆蓋方法 run(),我們在創建的 Thread 類的子類中重寫 run() ,加入線程所要執行的代碼即可。

下麵是一個例子:

 

 public class MyThread extends Thread

  {

   int count= 1, number;

   public MyThread(int num)

   {

    number = num;

    System.out.println

    ("創建線程 " + number);

   }

   public void run() {

    while(true) {

     System.out.println

      ("線程 " + number + ":計數 " + count);

     if(++count== 6) return;

    }

  }

  public static void main(String args[])

  {

   for(int i = 0;i 〈 5; i++) new MyThread(i+1).start();

  }

 }

 

 

  這種方法簡單明瞭,符合大家的習慣,但是,它也有一個很大的缺點,那就是如果我們的類已經從一個類繼承(如小程式必須繼承自 Applet 類),則無法再繼承 Thread 類,這時如果我們又不想建立一個新的類,應該怎麼辦呢?

 

  我們不妨來探索一種新的方法:我們不創建Thread類的子類,而是直接使用它,那麼我們只能將我們的方法作為參數傳遞給 Thread 類的實例,有點類似回調函數。但是 Java 沒有指針,我們只能傳遞一個包含這個方法的類的實例。

 

  那麼如何限制這個類必須包含這一方法呢?當然是使用介面!(雖然抽象類也可滿足,但是需要繼承,而我們之所以要採用這種新方法,不就是為了避免繼承帶來的限制嗎?)

 

  Java 提供了介面 java.lang.Runnable 來支持這種方法。

 

方法二:實現 Runnable 介面

  Runnable介面只有一個方法run(),我們聲明自己的類實現Runnable介面並提供這一方法,將我們的線程代碼寫入其中,就完成了這一部分的任務。但是Runnable介面並沒有任何對線程的支持,我們還必須創建Thread類的實例,這一點通過Thread類的構造函數 public Thread(Runnable target);來實現。下麵是一個例子:

 

  

public class MyThread implements Runnable

  {

   int count= 1, number;

   public MyThread(int num)

   {

    number = num;

    System.out.println("創建線程 " + number);

   }

   public void run()

   {

    while(true)

    {

     System.out.println

     ("線程 " + number + ":計數 " + count);

     if(++count== 6) return;

    }

   }

   public static void main(String args[])

   {

    for(int i = 0; i 〈 5;i++) new Thread(new MyThread(i+1)).start();

   }

  }

 

 

 

  

  嚴格地說,創建Thread子類的實例也是可行的,但是必須註意的是,該子類必須沒有覆蓋 Thread類的 run 方法,否則該線程執行的將是子類的 run 方法,而不是我們用以實現Runnable 介面的類的 run 方法,對此大家不妨試驗一下。

 

  使用 Runnable 介面來實現多線程使得我們能夠在一個類中包容所有的代碼,有利於封裝,它的缺點在於,我們只能使用一套代碼,若想創建多個線程並使各個線程執行不同的代碼,則仍必須額外創建類,如果這樣的話,在大多數情況下也許還不如直接用多個類分別繼承 Thread 來得緊湊。

 

3、使用ExecutorServiceCallableFuture實現有返回結果的多線程


ExecutorServiceCallableFuture這個對象實際上都是屬於Executor框架中的功能類。想要詳細瞭解Executor框架的可以訪問http://www.javaeye.com/topic/366591 ,這裡面對該框架做了很詳細的解釋。返回結果的線程是在JDK1.5中引入的新特征,確實很實用,有了這種特征我就不需要再為了得到返回值而大費周折了,而且即便實現了也可能漏洞百出。


可返回值的任務必須實現Callable介面,類似的,無返回值的任務必須Runnable介面。執行Callable任務後,可以獲取一個Future的對象,在該對象上調用get就可以獲取到Callable任務返回的Object了,再結合線程池介面ExecutorService就可以實現傳說中有返回結果的多線程了。下麵提供了一個完整的有返回結果的多線程測試例子,在JDK1.5下驗證過沒問題可以直接使用。代碼如下:

 

import java.util.concurrent.*;  

import java.util.Date;  

import java.util.List;  

import java.util.ArrayList;  

  

  

@SuppressWarnings("unchecked")  

public class Test {  

public static void main(String[] args) throws ExecutionException,  

    InterruptedException {  

   System.out.println("----程式開始運行----");  

   Date date1 = new Date();  

  

   int taskSize = 5;  

   // 創建一個線程池  

   ExecutorService pool = Executors.newFixedThreadPool(taskSize);  

   // 創建多個有返回值的任務  

   List list = new ArrayList();  

   for (int i = 0; i < taskSize; i++) {  

    Callable c = new MyCallable(i + " ");  

    // 執行任務並獲取Future對象  

    Future f = pool.submit(c);  

    // System.out.println(">>>" + f.get().toString());  

    list.add(f);  

   }  

   // 關閉線程池  

   pool.shutdown();  

  

   // 獲取所有併發任務的運行結果  

   for (Future f : list) {  

    // 從Future對象上獲取任務的返回值,並輸出到控制台  

    System.out.println(">>>" + f.get().toString());  

   }  

  

   Date date2 = new Date();  

   System.out.println("----程式結束運行----,程式運行時間【"  

     + (date2.getTime() - date1.getTime()) + "毫秒】");  

}  

}  

  

class MyCallable implements Callable {  

private String taskNum;  

  

MyCallable(String taskNum) {  

   this.taskNum = taskNum;  

}  

  

public Object call() throws Exception {  

   System.out.println(">>>" + taskNum + "任務啟動");  

   Date dateTmp1 = new Date();  

   Thread.sleep(1000);  

   Date dateTmp2 = new Date();  

   long time = dateTmp2.getTime() - dateTmp1.getTime();  

   System.out.println(">>>" + taskNum + "任務終止");  

   return taskNum + "任務返回運行結果,當前任務時間【" + time + "毫秒】";  

}  

}  

 

 

代碼說明:
上述代碼中Executors類,提供了一系列工廠方法用於創先線程池,返回的線程池都實現了ExecutorService介面。


public static ExecutorService newFixedThreadPool(int nThreads) 
創建固定數目線程的線程池。


public static ExecutorService newCachedThreadPool() 
創建一個可緩存的線程池,調用execute 將重用以前構造的線程(如果線程可用)。如果現有線程沒有可用的,則創建一個新線程並添加到池中。終止並從緩存中移除那些已有 60 秒鐘未被使用的線程。


public static ExecutorService newSingleThreadExecutor() 
創建一個單線程化的Executor


public static ScheduledExecutorService newScheduledThreadPool(int corePoolSize) 
創建一個支持定時及周期性的任務執行的線程池,多數情況下可用來替代Timer類。

ExecutoreService提供了submit()方法,傳遞一個Callable,或Runnable,返回Future。如果Executor後臺線程池還沒有完成Callable的計算,這調用返回Future對象的get()方法,會阻塞直到計算完成。

 

綜上所述,以上方法各有千秋,大家可以靈活運用。更多Java面試題全集和Java技術乾貨分享,請點擊上海尚學堂官網獲取,還有Java300集等Java學習視頻可以領取。


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 效果展示: 源碼:https://github.com/ProsperLee/demo-h5-package ...
  • 背景:easyui在做上下佈局的時候,上面是數據列表,下麵是數據圖表。如下圖 需要在上下麵板右上角加上最大化按鈕,以便可以全屏顯示。邏輯就是當上面點擊最大化時候,隱藏下麵,主意:此時需要將下麵的div標題設置為空字元,否則後果就是列表中間有一道人員能力統計圖標題;當點擊恢復按鈕的時候,展示下麵圖表, ...
  • LieBrother原文 : "行為型模式:策略模式" 十一大行為型模式之五:策略模式。 簡介 姓名 :策略模式 英文名 :Strategy Pattern 價值觀 :集計謀於一身 個人介紹 : Define a family of algorithms,encapsulate each one,a ...
  • 一、前言 什麼是命令模式? 在軟體系統中,“行為請求者”與“行為實現者”通常呈現一種“緊耦合”。但在某些場合,比如要對行為進行“記錄、撤銷/重做、事務”等處理,這種無法抵禦變化的緊耦合是不合適的。在這種情況下,如何將“行為請求者”與“行為實現者”解耦?將一組行為抽象為對象,實現二者之間的松耦合,這就 ...
  • 1.代碼生成器: [正反雙向](單表、主表、明細表、樹形表,快速開發利器)+快速表單構建器freemaker模版技術 ,0個代碼不用寫,生成完整的一個模塊,帶頁面、建表sql腳本、處理類、service等完整模塊2.多數據源:(支持同時連接無數個資料庫,可以不同的模塊連接不同數的據庫)支持N個數據源 ...
  • 一、什麼是OCTO 定義: OCTO是美團的分散式服務通信框架及服務治理系統,屬於公司級基礎設施,目前尚未開源。 目標: 為公司所有業務提供統一的服務通信框架,使業務具備良好的服務運營能力,輕鬆實現服務註冊、服務自動發現、負載均衡、容錯、灰度發佈、調用數據可視化等,持續提升服務高可用性、服務運維效率 ...
  • LieBrother原文 : "行為型模式:責任鏈模式" 十一大行為型模式之四:責任鏈模式。 簡介 姓名 :責任鏈模式 英文名 :Chain of Responsibility Pattern 價值觀 :責任歸我 個人介紹 : Avoid coupling the sender of a reque ...
  • 定義:裝飾模式是在不必改變原類文件和使用繼承的情況下,動態的擴展一個對象的功能。它是通過創建一個包裝對象,也就是裝飾來包裹真實的對象。 裝飾器模式是為已有功能添加更多功能的一種方式,就增加功能來說,裝飾器模式比通過生成子類更為靈活。該模式通過將裝飾的功能放在單獨的類中,並讓這些類包含了需要進行裝飾的 ...
一周排行
    -Advertisement-
    Play Games
  • 前言 本文介紹一款使用 C# 與 WPF 開發的音頻播放器,其界面簡潔大方,操作體驗流暢。該播放器支持多種音頻格式(如 MP4、WMA、OGG、FLAC 等),並具備標記、實時歌詞顯示等功能。 另外,還支持換膚及多語言(中英文)切換。核心音頻處理採用 FFmpeg 組件,獲得了廣泛認可,目前 Git ...
  • OAuth2.0授權驗證-gitee授權碼模式 本文主要介紹如何筆者自己是如何使用gitee提供的OAuth2.0協議完成授權驗證並登錄到自己的系統,完整模式如圖 1、創建應用 打開gitee個人中心->第三方應用->創建應用 創建應用後在我的應用界面,查看已創建應用的Client ID和Clien ...
  • 解決了這個問題:《winForm下,fastReport.net 從.net framework 升級到.net5遇到的錯誤“Operation is not supported on this platform.”》 本文內容轉載自:https://www.fcnsoft.com/Home/Sho ...
  • 國內文章 WPF 從裸 Win 32 的 WM_Pointer 消息獲取觸摸點繪製筆跡 https://www.cnblogs.com/lindexi/p/18390983 本文將告訴大家如何在 WPF 裡面,接收裸 Win 32 的 WM_Pointer 消息,從消息裡面獲取觸摸點信息,使用觸摸點 ...
  • 前言 給大家推薦一個專為新零售快消行業打造了一套高效的進銷存管理系統。 系統不僅具備強大的庫存管理功能,還集成了高性能的輕量級 POS 解決方案,確保頁面載入速度極快,提供良好的用戶體驗。 項目介紹 Dorisoy.POS 是一款基於 .NET 7 和 Angular 4 開發的新零售快消進銷存管理 ...
  • ABP CLI常用的代碼分享 一、確保環境配置正確 安裝.NET CLI: ABP CLI是基於.NET Core或.NET 5/6/7等更高版本構建的,因此首先需要在你的開發環境中安裝.NET CLI。這可以通過訪問Microsoft官網下載並安裝相應版本的.NET SDK來實現。 安裝ABP ...
  • 問題 問題是這樣的:第三方的webapi,需要先調用登陸介面獲取Cookie,訪問其它介面時攜帶Cookie信息。 但使用HttpClient類調用登陸介面,返回的Headers中沒有找到Cookie信息。 分析 首先,使用Postman測試該登陸介面,正常返回Cookie信息,說明是HttpCli ...
  • 國內文章 關於.NET在中國為什麼工資低的分析 https://www.cnblogs.com/thinkingmore/p/18406244 .NET在中國開發者的薪資偏低,主要因市場需求、技術棧選擇和企業文化等因素所致。歷史上,.NET曾因微軟的閉源策略發展受限,儘管後來推出了跨平臺的.NET ...
  • 在WPF開發應用中,動畫不僅可以引起用戶的註意與興趣,而且還使軟體更加便於使用。前面幾篇文章講解了畫筆(Brush),形狀(Shape),幾何圖形(Geometry),變換(Transform)等相關內容,今天繼續講解動畫相關內容和知識點,僅供學習分享使用,如有不足之處,還請指正。 ...
  • 什麼是委托? 委托可以說是把一個方法代入另一個方法執行,相當於指向函數的指針;事件就相當於保存委托的數組; 1.實例化委托的方式: 方式1:通過new創建實例: public delegate void ShowDelegate(); 或者 public delegate string ShowDe ...