国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 編程 > Java > 正文

詳解三種java實現多線程的方式

2019-11-26 15:01:53
字體:
來源:轉載
供稿:網友

java中實現多線程的方法有兩種:繼承Thread類和實現runnable接口

1.繼承Thread類,重寫父類run()方法

 public class thread1 extends Thread {     public void run() {        for (int i = 0; i < 10000; i++) {            System.out.println("我是線程"+this.getId());        }    }     public static void main(String[] args) {        thread1 th1 = new thread1();        thread1 th2 = new thread1();        th1.run();        th2.run();    }   }

run()方法只是普通的方法,是順序執行的,即th1.run()執行完成后才執行th2.run(),這樣寫只用一個主線程。多線程就失去了意義,所以應該用start()方法來啟動線程,start()方法會自動調用run()方法。上述代碼改為:

 public class thread1 extends Thread {         public void run() {        for (int i = 0; i < 10000; i++) {            System.out.println("我是線程"+this.getId());        }    }     public static void main(String[] args) {        thread1 th1 = new thread1();        thread1 th2 = new thread1();        th1.start();        th2.start();    }}

通過start()方法啟動一個新的線程。這樣不管th1.start()調用的run()方法是否執行完,都繼續執行th2.start()如果下面有別的代碼也同樣不需要等待th2.start()執行完成,而繼續執行。(輸出的線程id是無規則交替輸出的)

2.實現runnable接口

public class thread2 implements Runnable {     public String ThreadName;         public thread2(String tName){        ThreadName = tName;    }              public void run() {        for (int i = 0; i < 10000; i++) {            System.out.println(ThreadName);        }    }         public static void main(String[] args) {        thread2 th1 = new thread2("線程A");        thread2 th2 = new thread2("線程B");        th1.run();        th2.run();    }}

和Thread的run方法一樣Runnable的run只是普通方法,在main方法中th2.run()必須等待th1.run()執行完成后才能執行,程序只用一個線程。要多線程的目的,也要通過Thread的start()方法(注:runnable是沒有start方法)。上述代碼修改為:

public class thread2 implements Runnable {     public String ThreadName;         public thread2(String tName){        ThreadName = tName;    }              public void run() {        for (int i = 0; i < 10000; i++) {            System.out.println(ThreadName);        }    }         public static void main(String[] args) {        thread2 th1 = new thread2("線程A");        thread2 th2 = new thread2("Thread-B");        Thread myth1 = new Thread(th1);        Thread myth2 = new Thread(th2);        myth1.start();        myth2.start();    }}

3.使用ExecutorService、Callable、Future實現有返回結果的多線程(JDK5.0以后)
可返回值的任務必須實現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<Future> list = new ArrayList<Future>();   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<Object> { 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多線程的2種方式,runable是接口,thread是類,runnable只提供一個run方法,建議使用runable實現 java多線程,不管如何,最終都需要通過thread.start()來使線程處于可運行狀態。第三種方法是聽群里的兄弟們介紹的,所以就百度補上了。

以上就是本文的全部內容,希望對大家的學習有所幫助。

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
主站蜘蛛池模板: 邵东县| 海晏县| 玉山县| 苗栗市| 永登县| 土默特右旗| 米脂县| 平昌县| 石城县| 吉林省| 水富县| 凌海市| 勐海县| 比如县| 滨海县| 修武县| 亳州市| 长岛县| 大庆市| 鄄城县| 永德县| 靖边县| 甘洛县| 横峰县| 勐海县| 洮南市| 西宁市| 徐汇区| 宁城县| 平武县| 海兴县| 田阳县| 博客| 泰宁县| 鄄城县| 晋宁县| 平罗县| 衡东县| 古浪县| 如皋市| 临邑县|