本文介紹 C# 的 Task,包含範例程式碼。
為什麼使用 Task
- 提供比 Thread 更多的控制能力,如等候 (Wait)、取消 (Cancel)、接續 (ContinueWith) 以及錯誤處理 (exception handling)。不必再透過執行緒手動控制。
- Task 會被排入 ThreadPool,並在幕後提供負載平衡,以最大化輸出結果。
如何建立 Task 並等待至完成
和 Thread 類似,在建構式傳入要執行的方法,使用 Start() 開始執行工作,並使用 Wait() 等待工作完成。
int n;
private void AddN(){
for(int i = 0;i < 50;i++){
n = n + 1;
}
}
public void Run(){
Task task = new Task(AddN);
task.Start();
task.Wait();
Console.WriteLine(n.ToString());
}
另可參考 明確建立和執行工作 - Microsoft Docs 裡的範例。這段程式碼透過 Lambda 運算式執行,呼叫要執行的方法。
取消 (中斷) Task 的執行
- 建立一個
AddNWithCancel()的方法,傳入CancellationToken。方法內檢查 Token 的IsCancellationRequested,即取消被呼叫時,使用 Token 的ThrowIfCancellationRequested()拋出例外。 - 在執行這個方法前,建立
CancellationTokenSource和CancellationToken物件。使用Task.Run()執行工作,並使用 await 等待工作完成。由於我們在Task.Run()執行後,立刻呼叫 Token 的取消 (Cancel()) 方法,因此在底下的 try…catch… 區塊內,會接收到例外,並顯示「工作已取消」訊息。
private void AddNWithCancel(CancellationToken cancelToken){
int a = 0;
for(int i = 0;i < 1000;i++){
a = a + 1;
if(cancelToken.IsCancellationRequested){
cancelToken.ThrowIfCancellationRequested();
}
}
}
public async void Run(){
var tokenSource = new CancellationTokenSource();
CancellationToken cancelToken = tokenSource.Token;
var task2 = Task.Run(()=> AddNWithCancel(cancelToken), cancelToken);
tokenSource.Cancel();
try{
await task2;
}
catch(OperationCanceledException e){
Console.WriteLine("工作已取消");
}
}
也請參考 工作取消 - Microsoft Docs 裡的程式碼。與上方例子類似,拋出 OperationCanceledException 例外,然後結束 Task 程式的執行。
C# 學習筆記:多執行緒 (5) - 工作の取消和逾時 - Huan-Lin 學習筆記 的範例程式碼,則只要判斷 CancellationToken 內的 IsCancellationRequested 是否為 True,再適當的撰寫程式碼,結束 Task 即可,不透過拋出例外的方式取消 Task 的執行。
