本文介紹 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 的執行。