如何取消.net后臺(tái)線(xiàn)程的執(zhí)行
在使用多線(xiàn)程模型進(jìn)行編程時(shí),經(jīng)常遇到的問(wèn)題之一是,當(dāng)我們關(guān)閉前臺(tái)的UI線(xiàn)程時(shí),后臺(tái)的輔助線(xiàn)程仍然處于活動(dòng)狀態(tài),從而導(dǎo)致整個(gè)應(yīng)用程序無(wú)法正常退出。這時(shí)我們需要一種較安全的方式來(lái)結(jié)束后臺(tái)線(xiàn)程的運(yùn)行,這樣我們可以隨時(shí)結(jié)束后臺(tái)線(xiàn)程的運(yùn)行,并且在線(xiàn)程結(jié)束時(shí)進(jìn)行相應(yīng)的資源清理工作(例如將內(nèi)存數(shù)據(jù)寫(xiě)入硬盤(pán))。.net框架提供了一些工具來(lái)實(shí)現(xiàn)該功能。
目錄
IsBackground屬性
Abort方法
輪循方式
取消阻塞的線(xiàn)程
IsBackgound屬性
Thread類(lèi)提供了IsBackground屬性,當(dāng)線(xiàn)程的IsBackground屬性被設(shè)置為true時(shí),表示此線(xiàn)程為后臺(tái)工作線(xiàn)程。當(dāng)一個(gè)應(yīng)用程序結(jié)束時(shí),它的所有后臺(tái)線(xiàn)程會(huì)自動(dòng)的被結(jié)束執(zhí)行。如果你有一個(gè)后臺(tái)線(xiàn)程偵聽(tīng)Socket連接,并且正在被阻塞,那么這時(shí)候通過(guò)設(shè)置線(xiàn)程的IsBackground屬性為T(mén)rue,使它自動(dòng)隨應(yīng)用程序的結(jié)束而結(jié)束是比較合適的。但在這種情況下,線(xiàn)程會(huì)靜悄悄的結(jié)束,它不會(huì)引發(fā)任何異常,你的線(xiàn)程沒(méi)有機(jī)會(huì)執(zhí)行一些需要的清理代碼。例如,內(nèi)存中的數(shù)據(jù)可能會(huì)來(lái)不及寫(xiě)入磁盤(pán),從而造成丟失數(shù)據(jù)。
Abort方法
可以調(diào)用Thread類(lèi)的Abort方法來(lái)強(qiáng)制終制線(xiàn)程。上調(diào)用此方法時(shí),線(xiàn)程上引發(fā)ThreadAbortException,并導(dǎo)至線(xiàn)程終結(jié),通過(guò)捕獲該異常,可以執(zhí)行一些資源清理代碼。但這種模式也有一些問(wèn)題,主要是難以知道線(xiàn)程上的代碼執(zhí)行到什么地方,所有相應(yīng)的資源清理代碼也難以編寫(xiě)??偟膩?lái)說(shuō)這是一種比較粗暴的終止線(xiàn)程執(zhí)行的方法,通常來(lái)說(shuō)是不推薦使用的。
輪循方式
如果后臺(tái)線(xiàn)程將執(zhí)行一個(gè)很長(zhǎng)的計(jì)算,那么可以將計(jì)算隔成若干小段,并經(jīng)常檢查是否需要取消線(xiàn)程。.NET框架提供了CancellationTokenSource類(lèi)來(lái)作為線(xiàn)程取消的統(tǒng)一模式。例如:
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("請(qǐng)求取消線(xiàn)程的執(zhí)行");
cts.Cancel();
break;
}
}
Console.ReadLine();
}
private static void ThreadWork(object state)
{
CancellationToken cancellationToken = (CancellationToken)state;
while (true)
{
// 檢查是否取消
if(cancellationToken.IsCancellationRequested)
{
Console.WriteLine("線(xiàn)程已經(jīng)取消了");
Console.WriteLine("線(xiàn)程的資源已經(jīng)清理完成。");
break;
}
// 模擬工作
Thread.SpinWait(500000);
Console.WriteLine("我還在工作。");
}
}
}
取消阻塞的線(xiàn)程
上面的示例中,后臺(tái)線(xiàn)程會(huì)長(zhǎng)時(shí)間進(jìn)行計(jì)算,但更多的時(shí)候,線(xiàn)程會(huì)由于等待某個(gè)事件,從而進(jìn)入阻塞狀態(tài)。這個(gè)時(shí)候,實(shí)際上線(xiàn)程已經(jīng)不再執(zhí)行狀態(tài)了,很明顯,它沒(méi)有機(jī)會(huì)去檢查取消標(biāo)志。 那么,該如何解決這個(gè)問(wèn)題呢?CancellationToken的WaitHandle屬性提供了解答。WaitHandle類(lèi)有一個(gè)靜態(tài)方法WaitAny,它可以同時(shí)等待多個(gè)事件,當(dāng)多個(gè)事件中的任意一個(gè)有效時(shí),線(xiàn)程都會(huì)從阻塞狀態(tài)中返回??梢愿鶕?jù)WaitAny方法的返回值來(lái)判斷發(fā)生了什么事件,從而相應(yīng)的執(zhí)行代碼。例子:
public class Example
{
private static int Value;
public static void Main()
{
var autoResetEvent = new AutoResetEvent(false);
var cts = new CancellationTokenSource();
var state = new { ValueAvailableEvent = autoResetEvent, CancellationToken = cts.Token };
var threadConsumer = new Thread(ConsumerThreadWork);
var threadProducter = new Thread(ProducterThreadWork);
threadConsumer.Start(state);
threadProducter.Start(state);
while (true)
{
if (Console.ReadKey().KeyChar == 'c')
{
Console.WriteLine("請(qǐng)求取消線(xiàn)程的執(zhí)行");
cts.Cancel();
break;
}
}
Console.ReadLine();
}
public static void ProducterThreadWork(dynamic state)
{
var valueAvailableEvent = (AutoResetEvent)state.ValueAvailableEvent;
var cancellationToken = (CancellationToken)state.CancellationToken;
var rand = new Random();
while (!cancellationToken.IsCancellationRequested)
{
Value = rand.Next();
Console.WriteLine("\r\n產(chǎn)生一個(gè)值{0}", Value);
valueAvailableEvent.Set();
Thread.Sleep(500);
}
Console.WriteLine("生產(chǎn)者線(xiàn)程被取消。");
}
public static void ConsumerThreadWork(dynamic state)
{
var valueAvailableEvent = (AutoResetEvent)state.ValueAvailableEvent;
var cancellationToken = (CancellationToken)state.CancellationToken;
var events = new[] { valueAvailableEvent, cancellationToken.WaitHandle };
while (true)
{
var eventIndex = WaitHandle.WaitAny(events);
// 處理數(shù)據(jù)
if (eventIndex == 0)
{
Console.WriteLine("處理值{0}。", Value);
}
// 處理取消事件
else if (eventIndex == 1)
{
Console.WriteLine("消費(fèi)者線(xiàn)程被取消。");
break;
}
}
}
}
在上面的例子中,有三個(gè)線(xiàn)程,分別是UI線(xiàn)程,生產(chǎn)者線(xiàn)程和消費(fèi)者線(xiàn)程。其中生產(chǎn)者線(xiàn)程每隔一秒產(chǎn)生一個(gè)有效數(shù)值,并將數(shù)據(jù)保存到Value字段中,而消費(fèi)者線(xiàn)程等待值的產(chǎn)生,這個(gè)等待的過(guò)程是阻塞的。消費(fèi)都線(xiàn)程通過(guò)WaitHandle.WaitAny方法來(lái)同時(shí)等待值有效事件或者取消事件,當(dāng)任意一個(gè)事件有效時(shí),線(xiàn)程都將繼續(xù),并且通過(guò)返回的值來(lái)判斷發(fā)生的事件,并作相應(yīng)的處理。
總結(jié)
多線(xiàn)程模型中的線(xiàn)程取消問(wèn)題還是比較復(fù)雜的。Thread.IsBackground屬性提供了在前臺(tái)線(xiàn)程結(jié)束后自動(dòng)結(jié)束線(xiàn)程的方法。Thread.Abort方法提供了一種“粗暴”的結(jié)束線(xiàn)程的方法。CancellationTokenSource類(lèi)則是線(xiàn)程取消的標(biāo)準(zhǔn)模式,我們應(yīng)當(dāng)更多的使用這種模式。文章寫(xiě)的不多,基本是字?jǐn)?shù)不夠,代碼來(lái)湊,大家伙將就的看看吧。
相關(guān)文章
vscode extension插件開(kāi)發(fā)詳解
這篇文章主要介紹了vscode extension插件開(kāi)發(fā),小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2019-05-05
asp.net 自定義控件實(shí)現(xiàn)無(wú)刷新上傳圖片,立即顯示縮略圖,保存圖片縮略圖
自定義控件實(shí)現(xiàn)無(wú)刷新上傳圖片,立即顯示縮略圖,保存圖片縮略圖2010-01-01
擴(kuò)展 Entity Framework支持復(fù)雜的過(guò)濾條件(多個(gè)關(guān)鍵字模糊匹配)
之前遇到一個(gè)棘手的Linq to EF查詢(xún)的技術(shù)問(wèn)題,現(xiàn)有產(chǎn)品表Product,需要根據(jù)多個(gè)關(guān)鍵字模糊匹配產(chǎn)品名稱(chēng), 現(xiàn)將解決方案分享出來(lái),按興趣的朋友可以參考下2012-12-12
asp.net操作javascript:confirm返回值的兩種方式
asp.net操作javascript:confirm返回值分為兩種,不使用ajax、使用了ajax,不使用ajax,可以用StringBuilder來(lái)完成2014-09-09
.net中自定義錯(cuò)誤頁(yè)面的實(shí)現(xiàn)方法
這篇文章主要給大家介紹了關(guān)于.net中自定義錯(cuò)誤頁(yè)面實(shí)現(xiàn)的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2018-06-06
asp.net 虛擬主機(jī)時(shí)常出現(xiàn)MAC驗(yàn)證失敗錯(cuò)誤之解決方法
驗(yàn)證視圖狀態(tài) MAC 失敗。如果此應(yīng)用程序由網(wǎng)絡(luò)場(chǎng)或群集承載,請(qǐng)確保 <machineKey> 配置指定了相同的 validationKey 和驗(yàn)證算法。不能在群集中使用 AutoGenerate。2009-05-05
ASP.NET中DES加密與解密MD5加密幫助類(lèi)的實(shí)現(xiàn)代碼
這篇文章主要介紹了ASP.NET中DES加密與解密MD5加密幫助類(lèi)的實(shí)例代碼,非常不錯(cuò),具有參考借鑒價(jià)值,需要的朋友可以參考下2017-07-07
ASP.NET 返回隨機(jī)數(shù)實(shí)現(xiàn)代碼
ASP.NET返回隨機(jī)數(shù),需要的朋友可以參考下。2009-11-11
Asp.Net Core基于JWT認(rèn)證的數(shù)據(jù)接口網(wǎng)關(guān)實(shí)例代碼
這篇文章主要給大家介紹了關(guān)于Asp.Net Core基于JWT認(rèn)證的數(shù)據(jù)接口網(wǎng)關(guān)的相關(guān)資料,文中通過(guò)示例代碼以及圖文介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者使用Asp.net Core具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2019-03-03

