国产99久久精品_欧美日本韩国一区二区_激情小说综合网_欧美一级二级视频_午夜av电影_日本久久精品视频

最新文章專題視頻專題問答1問答10問答100問答1000問答2000關鍵字專題1關鍵字專題50關鍵字專題500關鍵字專題1500TAG最新視頻文章推薦1 推薦3 推薦5 推薦7 推薦9 推薦11 推薦13 推薦15 推薦17 推薦19 推薦21 推薦23 推薦25 推薦27 推薦29 推薦31 推薦33 推薦35 推薦37視頻文章20視頻文章30視頻文章40視頻文章50視頻文章60 視頻文章70視頻文章80視頻文章90視頻文章100視頻文章120視頻文章140 視頻2關鍵字專題關鍵字專題tag2tag3文章專題文章專題2文章索引1文章索引2文章索引3文章索引4文章索引5123456789101112131415文章專題3
問答文章1 問答文章501 問答文章1001 問答文章1501 問答文章2001 問答文章2501 問答文章3001 問答文章3501 問答文章4001 問答文章4501 問答文章5001 問答文章5501 問答文章6001 問答文章6501 問答文章7001 問答文章7501 問答文章8001 問答文章8501 問答文章9001 問答文章9501
當前位置: 首頁 - 科技 - 知識百科 - 正文

如何取消.net后臺線程的執(zhí)行

來源:懂視網 責編:小采 時間:2020-11-27 22:42:15
文檔

如何取消.net后臺線程的執(zhí)行

如何取消.net后臺線程的執(zhí)行:介紹 在使用多線程模型進行編程時,經常遇到的問題之一是,當我們關閉前臺的UI線程時,后臺的輔助線程仍然處于活動狀態(tài),從而導致整個應用程序無法正常退出。這時我們需要一種較安全的方式來結束后臺線程的運行,這樣我們可以隨時結束后臺線程的運行,并且在
推薦度:
導讀如何取消.net后臺線程的執(zhí)行:介紹 在使用多線程模型進行編程時,經常遇到的問題之一是,當我們關閉前臺的UI線程時,后臺的輔助線程仍然處于活動狀態(tài),從而導致整個應用程序無法正常退出。這時我們需要一種較安全的方式來結束后臺線程的運行,這樣我們可以隨時結束后臺線程的運行,并且在

介紹
在使用多線程模型進行編程時,經常遇到的問題之一是,當我們關閉前臺的UI線程時,后臺的輔助線程仍然處于活動狀態(tài),從而導致整個應用程序無法正常退出。這時我們需要一種較安全的方式來結束后臺線程的運行,這樣我們可以隨時結束后臺線程的運行,并且在線程結束時進行相應的資源清理工作(例如將內存數(shù)據(jù)寫入硬盤)。.net框架提供了一些工具來實現(xiàn)該功能。

目錄
IsBackground屬性
Abort方法
輪循方式
取消阻塞的線程
IsBackgound屬性
Thread類提供了IsBackground屬性,當線程的IsBackground屬性被設置為true時,表示此線程為后臺工作線程。當一個應用程序結束時,它的所有后臺線程會自動的被結束執(zhí)行。如果你有一個后臺線程偵聽Socket連接,并且正在被阻塞,那么這時候通過設置線程的IsBackground屬性為True,使它自動隨應用程序的結束而結束是比較合適的。但在這種情況下,線程會靜悄悄的結束,它不會引發(fā)任何異常,你的線程沒有機會執(zhí)行一些需要的清理代碼。例如,內存中的數(shù)據(jù)可能會來不及寫入磁盤,從而造成丟失數(shù)據(jù)。

Abort方法
可以調用Thread類的Abort方法來強制終制線程。上調用此方法時,線程上引發(fā)ThreadAbortException,并導至線程終結,通過捕獲該異常,可以執(zhí)行一些資源清理代碼。但這種模式也有一些問題,主要是難以知道線程上的代碼執(zhí)行到什么地方,所有相應的資源清理代碼也難以編寫。總的來說這是一種比較粗暴的終止線程執(zhí)行的方法,通常來說是不推薦使用的。

輪循方式
如果后臺線程將執(zhí)行一個很長的計算,那么可以將計算隔成若干小段,并經常檢查是否需要取消線程。.NET框架提供了CancellationTokenSource類來作為線程取消的統(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("請求取消線程的執(zhí)行");
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("我還在工作。");
}
}
}

取消阻塞的線程
上面的示例中,后臺線程會長時間進行計算,但更多的時候,線程會由于等待某個事件,從而進入阻塞狀態(tài)。這個時候,實際上線程已經不再執(zhí)行狀態(tài)了,很明顯,它沒有機會去檢查取消標志。 那么,該如何解決這個問題呢?CancellationToken的WaitHandle屬性提供了解答。WaitHandle類有一個靜態(tài)方法WaitAny,它可以同時等待多個事件,當多個事件中的任意一個有效時,線程都會從阻塞狀態(tài)中返回。可以根據(jù)WaitAny方法的返回值來判斷發(fā)生了什么事件,從而相應的執(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("請求取消線程的執(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產生一個值{0}", Value);
valueAvailableEvent.Set();
Thread.Sleep(500);
}

Console.WriteLine("生產者線程被取消。");
}

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("消費者線程被取消。");
break;
}
}
}
}

在上面的例子中,有三個線程,分別是UI線程,生產者線程和消費者線程。其中生產者線程每隔一秒產生一個有效數(shù)值,并將數(shù)據(jù)保存到Value字段中,而消費者線程等待值的產生,這個等待的過程是阻塞的。消費都線程通過WaitHandle.WaitAny方法來同時等待值有效事件或者取消事件,當任意一個事件有效時,線程都將繼續(xù),并且通過返回的值來判斷發(fā)生的事件,并作相應的處理。

總結
多線程模型中的線程取消問題還是比較復雜的。Thread.IsBackground屬性提供了在前臺線程結束后自動結束線程的方法。Thread.Abort方法提供了一種“粗暴”的結束線程的方法。CancellationTokenSource類則是線程取消的標準模式,我們應當更多的使用這種模式。文章寫的不多,基本是字數(shù)不夠,代碼來湊,大家伙將就的看看吧。

聲明:本網頁內容旨在傳播知識,若有侵權等問題請及時與本網聯(lián)系,我們將在第一時間刪除處理。TEL:177 7030 7066 E-MAIL:11247931@qq.com

文檔

如何取消.net后臺線程的執(zhí)行

如何取消.net后臺線程的執(zhí)行:介紹 在使用多線程模型進行編程時,經常遇到的問題之一是,當我們關閉前臺的UI線程時,后臺的輔助線程仍然處于活動狀態(tài),從而導致整個應用程序無法正常退出。這時我們需要一種較安全的方式來結束后臺線程的運行,這樣我們可以隨時結束后臺線程的運行,并且在
推薦度:
標簽: 取消 關閉的 怎么
  • 熱門焦點

最新推薦

猜你喜歡

熱門推薦

專題
Top
主站蜘蛛池模板: 日韩福利一区 | 在线观看亚洲一区二区 | 日本v片免费一区二区三区 欧洲精品欧美精品 | 性欧美大战久久久久久久野外黑人 | 伊人一区 | 欧美v在线| 99热成人精品免费久久 | 91精品导航 | 在线视频一区二区 | 亚州色图欧美色图 | 久久免费国产精品一区二区 | 精品国产高清自在线一区二区三区 | 日韩午夜电影 | 国产精品亚洲色图 | 精品欧美一区二区三区 | 国产91精品对白露脸全集观看 | 日韩欧美高清 | 日韩在线第三页 | 久久黄色毛片 | 国产精品久久新婚兰兰 | 久久一区二区精品综合 | 国产麻豆a一级毛片爽爽影院 | 亚洲精品制服丝袜二区 | 一区二区三区四区日韩 | www蜜桃| 亚洲 欧美 日韩 在线 | 日韩欧美亚洲视频 | 亚洲专区欧美专区 | 国产区精品福利在线社区 | 在线免费观看国产 | 欧美日韩另类综合 | 日韩欧美在线视频 | 四虎精品 | 国产精品久久久久免费 | 亚洲欧洲高清 | 青青草原国产一区二区 | 亚洲一区精品伊人久久 | 国产成人久久综合二区 | 日韩欧美在线观看视频 | 北条麻妃国产九九九精品视频 | 国产v欧美v日韩在线观看 |