java中實現多線程的方法有兩種:繼承Thread類和實現runnable接口。
1.繼承Thread類,重寫父類run()方法
run()方法只是普通的方法,是順序執行的,即th1.run()執行完成後才執行th2.run(),這樣寫只用一個主線程。多線程就失去了意義,所以應該用start()方法來啓動線程,start()方法會自動調用run()方法。上述代碼改爲:
通過start()方法啓動一個新的線程。這樣不管th1.start()調用的run()方法是否執行完,都繼續執行th2.start()如果下面有別的代碼也同樣不需要等待th2.start()執行完成,而繼續執行。(輸出的線程id是無規則交替輸出的)
2.實現runnable接口
和Thread的run方法一樣Runnable的run只是普通方法,在main方法中th2.run()必須等待th1.run()執行完成後才能執行,程序只用一個線程。要多線程的目的,也要通過Thread的start()方法(注:runnable是沒有start方法)。上述代碼修改爲:
3.使用ExecutorService、Callable、Future實現有返回結果的多線程(JDK5.0以後)
可返回值的任務必須實現Callable接口,類似的,無返回值的任務必須Runnable接口。執行Callable任務後,可以獲取一個Future的對象,在該對象上調用get就可以獲取到Callable任務返回的Object了,再結合線程池接口ExecutorService就可以實現傳說中有返回結果的多線程了。下面提供了一個完整的有返回結果的多線程測試例子,在JDK1.5下驗證過沒問題可以直接使用。代碼如下:
代碼說明:
上述代碼中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()來使線程處於可運行狀態。第三種方法是聽羣裏的兄弟們介紹的,所以就百度補上了。