如何取消.net后臺線程的執行
2024-07-10 12:41:15
供稿:網友
介紹
在使用多線程模型進行編程時,經常遇到的問題之一是,當我們關閉前臺的UI線程時,后臺的輔助線程仍然處于活動狀態,從而導致整個應用程序無法正常退出。這時我們需要一種較安全的方式來結束后臺線程的運行,這樣我們可以隨時結束后臺線程的運行,并且在線程結束時進行相應的資源清理工作(例如將內存數據寫入硬盤)。.net框架提供了一些工具來實現該功能。
目錄
IsBackground屬性
Abort方法
輪循方式
取消阻塞的線程
IsBackgound屬性
Thread類提供了IsBackground屬性,當線程的IsBackground屬性被設置為true時,表示此線程為后臺工作線程。當一個應用程序結束時,它的所有后臺線程會自動的被結束執行。如果你有一個后臺線程偵聽Socket連接,并且正在被阻塞,那么這時候通過設置線程的IsBackground屬性為True,使它自動隨應用程序的結束而結束是比較合適的。但在這種情況下,線程會靜悄悄的結束,它不會引發任何異常,你的線程沒有機會執行一些需要的清理代碼。例如,內存中的數據可能會來不及寫入磁盤,從而造成丟失數據。
Abort方法
可以調用Thread類的Abort方法來強制終制線程。上調用此方法時,線程上引發ThreadAbortException,并導至線程終結,通過捕獲該異常,可以執行一些資源清理代碼。但這種模式也有一些問題,主要是難以知道線程上的代碼執行到什么地方,所有相應的資源清理代碼也難以編寫。總的來說這是一種比較粗暴的終止線程執行的方法,通常來說是不推薦使用的。
輪循方式
如果后臺線程將執行一個很長的計算,那么可以將計算隔成若干小段,并經常檢查是否需要取消線程。.NET框架提供了CancellationTokenSource類來作為線程取消的統一模式。例如:
代碼如下:
public class Example
{
public static void Main()
{
CancellationTokenSource cts = new CancellationTokenSource();
var thread = new Thread(ThreadWork);
thread.Start(cts.Token);
while (true)
{
if(Console.ReadKey().KeyChar == 'c')
{
Console.WriteLine("請求取消線程的執行");
cts.Cancel();
break;
}
}
Console.ReadLine();
}
private static void ThreadWork(object state)
{
CancellationToken cancellationToken = (CancellationToken)state;
while (true)
{
// 檢查是否取消
if(cancellationToken.IsCancellationRequested)
{
Console.WriteLine("線程已經取消了");
Console.WriteLine("線程的資源已經清理完成。");
break;
}
// 模擬工作
Thread.SpinWait(500000);
Console.WriteLine("我還在工作。");
}
}
}
取消阻塞的線程
上面的示例中,后臺線程會長時間進行計算,但更多的時候,線程會由于等待某個事件,從而進入阻塞狀態。這個時候,實際上線程已經不再執行狀態了,很明顯,它沒有機會去檢查取消標志。 那么,該如何解決這個問題呢?CancellationToken的WaitHandle屬性提供了解答。WaitHandle類有一個靜態方法WaitAny,它可以同時等待多個事件,當多個事件中的任意一個有效時,線程都會從阻塞狀態中返回??梢愿鶕aitAny方法的返回值來判斷發生了什么事件,從而相應的執行代碼。例子: