4.1.5 系統管理調度——Executor介面
Executor介面(java.util.concurrent包的一部分)就代表了一個根據某種策略來執行Runnable執行個體的對象,其中可能包括了排隊和調度的細節,或如何選擇要執行的任務。Executor介面只定義了一個方法:
interface Executor {void execute(Runnable task);}
Java提供了大量的內建Executor介面實現,它們都可以簡單方便地使用,也可以進行擴充性的配置。其中一些還提供了處理維護線程等繁瑣細節的功能。例如,如果一個線程因為未捕獲的異常或其他故障停止,它們就自動建立一個新的線程來替換原來的線程。
ExecutorService介面繼承於Executor介面,並提供了一個更進階的工具來關閉伺服器,包括正常的關閉和突然的關閉。ExecutorService還允許在完成任務後返回一個結果,這需要用到Callable介面,它和Runnable介面很像,只是多了一個傳回值。
我們可以通過調用Executors類的各種靜態Factory 方法來擷取ExecutorService執行個體。樣本程式TCPEchoServerExecutor.java示範了基本Executor工具的使用方法。
下面是詳細代碼
package com.suifeng.tcpip.chapter4;import java.io.IOException;import java.net.ServerSocket;import java.net.Socket;import java.util.concurrent.Executor;import java.util.concurrent.Executors;import java.util.logging.Logger;/** * 採用JDK內建線程池的方式 * @author Suifeng * */public class TCPEchoServerExecutors{public static void main(String[] args) throws IOException{if(args.length != 1){throw new IllegalArgumentException("Parameter:<Port>");}// 伺服器監聽連接埠int serverPort = Integer.parseInt(args[0]);ServerSocket serverSocket = new ServerSocket(serverPort);/* * 在使用一個實現了Runnable介面的執行個體調用它的execute()方法時, * 如果必要它將建立一個新的線程來處理任務。 * 然而,它首先會嘗試使用已有的線程。 * 如果一個線程空閑了60秒以上,則將移出線程池。 * */ Executor service = Executors.newCachedThreadPool();Logger logger = Logger.getLogger("practical");logger.info("Server is started!!!!");while(true){Socket socket = serverSocket.accept();service.execute(new EchoProtocol(socket,logger));}}}
啟動伺服器端,監聽39393連接埠
啟動用戶端
查看伺服器端
再次啟動用戶端
再次查看伺服器端
本次結果可以看出,兩次使用的是同一個線程進行處理。