對筆者近期剛閱讀完的Thinking In Java的12.2節簡單作個概述:java
本節的主要內容:編程
Runnable
接口、Callable
接口以及繼承 Thread
類,而且各自實現的優點Executor
(線程執行器),簡要介紹了 CachedThreadPoo
l、SingleThreadPool
以及 FixedThreadPool
三種基本的線程執行器Runnable
接口(無返回值的任務描述)以及 Callable
接口(有返回值的任務描述)Thread
對象的 基本操做方式: Thread t
=> t.yield()
t.start()
t.sleep()
t.join()
t.wait()
t.interrupt
( join
, wait
和interrupt
筆者沒有書寫特定的示例,在下方的博客資料中有詳細的更優秀的博客作了相關方面的描述)run()
方法中逃逸的異常(在Main線程中沒法try/catch到的)如下是筆者的筆記,對於Java中的進程的基本概念有不少知識還須要參考操做系統(如:線程的三種基本態:執行態、等待態、執行態以及補充的掛起態和激活態)。大部分筆者本身的理解都放在了代碼中,若是有疏漏的還期望dalao們私聊指點下筆者(小萌新一隻)!緩存
package cnboy.henu.xb; import java.util.concurrent.Callable; import java.util.concurrent.ExecutorService; import java.util.concurrent.Executors; import java.util.concurrent.ThreadFactory; /** * Thinking In Java 21.2 基本的線程進制 學習的樣例代碼 * @author Administrator * * 最重要要理清概念 「線程」與「任務」,任務須要線程的驅動,Runnable以及Callable接口所作的事情是描述一個任務,而Thread則是開啓一個線程,附着在任務上,驅動任務的執行 */ public class ExcutorTry { public static void main(String[] args) { // TODO Auto-generated method stub // // 執行器對象 提供中間層管理任務對象(線程) // /** // * 下面是使用的SingleTheadPool作的示例 // */ // ExecutorService exec1 = Executors.newSingleThreadExecutor(); // // 此對象緩存的線程對象僅只有一個,所以其它須要執行的任務會自動排隊等待執行 // for(int i=0;i<10;i++) { // exec1.execute(new TaskWithoutResult(i)); // } // exec1.shutdown(); // /** // * 下面是利用的 CachedThreadPool 作的示例 // */ // ExecutorService exec = Executors.newCachedThreadPool(); // // Future對象正如名字所示,能夠表明一個異步任務返回的對象並提供獲取其值的方式 // List<Future<String>> results = new ArrayList<Future<String>>(); // for(int i=0;i<10;i++) { // results.add(exec.submit(new TaskWithResult(i))); // } // // 關閉執行器,不讓新的任務添加進來 // exec.shutdown(); // for(Future<String> fs:results) { // try { // // 判斷fs是否已經執行完畢,也能夠不加,不加的狀況下會自動阻塞等待其執行完畢 // if(fs.isDone()) { // System.out.println(fs.get()); // } // }catch(InterruptedException e){ // e.printStackTrace(); // } catch (ExecutionException e) { // // TODO Auto-generated catch block // e.printStackTrace(); // } // } // // 這裏執行描述的任務的Runner其實是Main線程而不是launch對象 LiffOff launch = new LiffOff(10);launch.run(); Thread.yield(); // // 這裏執行描述的任務對象是線程 t ,此處纔是真正的多線程併發執行(main 線程以及 t 線程) // Thread t = new Thread(new LiffOff(10)); // t.start();System.out.println("Waiting for Liff Out!"); // // 這裏模仿了6個線程併發的執行狀況(main + 五個new Thread) // for(int i=0;i<5;i++) { // new Thread(new LiffOff(10)).start(); // } /** * Daemon線程 (守護線程,定義爲DaemonThread意味着此線程不屬於程序中不可或缺的一部分,常見的 GC線程是Daemon線程) * Daemon進程fork出來的子線程也會被自動設置爲Daemon線程 */ // // 將線程執行器設置爲後臺線程執行器(即建立執行任務的線程都是後臺線程),採用構造注入的方式將後臺線程Factory注入執行器中 // ExecutorService daemonExec = Executors.newCachedThreadPool(new DaemonThreadFactory()); // for(int i =0;i<5;i++) { // // 後臺執行 // daemonExec.execute(new TaskWithoutResult(i)); // } // daemonExec.shutdown(); // System.out.println("All Daemon Thread Started!"); // try { // // 等待一下Daemon 線程的執行(有趣的是能夠調節等待時間觀察Daemon線程執行與非後臺線程執行的關係,即非後臺線程執行完畢,Daemon線程也會同時退出) // //TimeUnit.SECONDS.sleep(10);// 後臺線程執行完畢 // TimeUnit.MICROSECONDS.sleep(1000);// 後臺線程來不及執行完畢 // } catch (InterruptedException e) { // // TODO Auto-generated catch block // e.printStackTrace(); // } /** * 使用繼承Thread類的方式實現多線程,缺點在於沒法再繼承其餘類,而實現Runnable接口任然能夠繼承其餘類 */ // for(int i=0;i<5;i++) { // new SimpleThread(); // } /** * 一般咱們在run()裏出現的異常(從線程中逃逸,向父線程傳播)若是不加以特殊處理(如使用UncaughtExceptionHandle),而只是簡單的在Main裏經過try/catch是沒辦法捕獲並處理異常的 * 所以咱們須要引入UncaughtExceptionHandle爲咱們的線程加上一道安全防線(使用在ThreadFactory、Executor) * */ // ExecutorService exceptionExec = // Executors.newCachedThreadPool(new HandleThreadFactory()); // exceptionExec.execute(new ExceptionThread()); // exceptionExec.shutdown(); } } /** * 不帶返回值的任務 * @author Administrator * */ class TaskWithoutResult implements Runnable{ private int id; public TaskWithoutResult(int id) { // TODO Auto-generated constructor stub this.id = id; System.out.println("Generate Runnable Task : id = "+id); } @Override public void run() { // TODO Auto-generated method stub System.out.println(String.format("Task Without Result => id = %d", id)); } } /** * 帶返回值的任務 實現的Callable接口 * @author Administrator * */ class TaskWithResult implements Callable<String>{ private int id=0; public TaskWithResult(int id) { // TODO Auto-generated constructor stub this.id = id; } /** * 注意不是Run方法 */ @Override public String call() throws Exception { // TODO Auto-generated method stub // 模仿下有任務的線程 return String.format("ask With Result => id = %d", id); } } /** * 這個任務描述是模擬的火箭發射的例子 * @author Administrator * */ class LiffOff implements Runnable{ protected int countDown =10; private static int taskCount = 0; private final int id=taskCount++; public LiffOff(int countDown) { // TODO Auto-generated constructor stub this.countDown = countDown; } public String Status() { return String.format("# %d (", id)+(countDown>0?countDown:"Liff Off!")+")"; } @Override public void run() { // 模擬火箭發射倒計時 while(countDown-- >0) { System.out.println(Status()); // 讓步 告訴線程調度器此線程核心工做已完成,能夠進行上下文切換了 /** * 與sleep不一樣的是,sleep是讓線程進入阻塞態,這時不管比線程高優先級仍是低優先級的線程均可以被執行 * 而yeild僅僅只是讓出此時間片,進入可執行態也就是它並不會等待一次完整的輪轉後再之執行,而是有可能隨時又進入執行態(可能剛yeild完畢又被調用了),所以它只能讓步與它同優先級或者是高優先級的線程 */ //Thread.yield(); /** * 換成sleep,能夠看到每一個線程都會按照時間輪轉進行輸出,而不是搶佔的輸出(會按照順序輸出) */ try { Thread.sleep(100); } catch (InterruptedException e) { // TODO Auto-generated catch block e.printStackTrace(); } } } } /** * 專門用於建立後臺進程的ThreadFactory * @author Administrator * */ class DaemonThreadFactory implements ThreadFactory{ @Override public Thread newThread(Runnable r) { // TODO Auto-generated method stub Thread t = new Thread(r); // 這裏設置線程爲後臺線程 t.setDaemon(true); return t; } } /** * 模擬一種實現線程的方式:繼承自Thread類 * @author Administrator * */ class SimpleThread extends Thread{ protected int countDown =5; private static int taskCount = 0; public SimpleThread() { // TODO Auto-generated constructor stub // 經過調用父類的構造函數,給線程名字賦值 super(Integer.toString(taskCount++)); // 在構造函數中便啓動線程 -- 不推薦的方式,僅爲了方便作示例 start(); } @Override public String toString() { // TODO Auto-generated method stub return "#"+getName()+"("+countDown+")"; } @Override public void run() { // TODO Auto-generated method stub while(true) { System.out.println(this); if(countDown-- ==0) { return; } } } } /** * 模擬一個在run()方法中拋出異常的Runnable對象 * @author Administrator * */ class ExceptionThread implements Runnable{ @Override public void run() { // TODO Auto-generated method stub // 模擬執行過程當中拋出異常 Thread t = Thread.currentThread(); System.out.println("run() by "+ t); System.out.println("eh = "+t.getUncaughtExceptionHandler()); throw new RuntimeException(); } } /** * 模擬一個用以處理逃逸的異常的UncaughtExceptionHandler對象 * @author Administrator * */ class MyUncaughtExceptionHandler implements Thread.UncaughtExceptionHandler{ /** * uncaughtException方法會在線程因未捕獲的異常而面臨死亡的時候被調用 */ @Override public void uncaughtException(Thread t, Throwable e) { // TODO Auto-generated method stub System.out.println("caught :"+e); } } /** * 模擬的一個爲線程設置異常處理的ThreadFactory * @author Administrator * */ class HandleThreadFactory implements ThreadFactory{ @Override public Thread newThread(Runnable r) { // TODO Auto-generated method stub // 記錄建立線程的過程 System.out.println(this+"\tcreating new Thread"); Thread t = new Thread(r); System.out.println("created \t" + t); // 給與線程UncaughtException處理器 t.setUncaughtExceptionHandler(new MyUncaughtExceptionHandler()); System.out.println("eh = "+t.getUncaughtExceptionHandler()); return t; } }
附:學習中借鑑的幾篇優秀的博客安全