国产片侵犯亲女视频播放_亚洲精品二区_在线免费国产视频_欧美精品一区二区三区在线_少妇久久久_在线观看av不卡

服務(wù)器之家:專注于服務(wù)器技術(shù)及軟件下載分享
分類導(dǎo)航

PHP教程|ASP.NET教程|Java教程|ASP教程|編程技術(shù)|正則表達(dá)式|C/C++|IOS|C#|Swift|Android|VB|R語言|JavaScript|易語言|vb.net|

服務(wù)器之家 - 編程語言 - ASP.NET教程 - ASP.NET 謹(jǐn)用 async/await

ASP.NET 謹(jǐn)用 async/await

2020-05-22 12:12Beck''s Blog ASP.NET教程

這篇文章主要介紹了ASP.NET 應(yīng)用程序?qū)嶋H使用過程中的一些總結(jié), 包括 異常捕獲 、 死鎖 、 應(yīng)用程序崩潰 ,實際使用過程中一不注意就可能掉坑里了

C# 5.0 引入 async/await 關(guān)鍵字,旨在簡化異步編程模型,拋去語法糖就是 Net4.0 的 Task + 狀態(tài)機(jī)。其實在處理異步編程使用 Task 還是挺簡單的,不過既然推出了新的語法糖,難免會嘗試一下,然而在使用中卻沒想象中那么單純。以下針對ASP.NET 應(yīng)用程序?qū)嶋H使用過程中的一些總結(jié), 包括 異常捕獲 、 死鎖 、 應(yīng)用程序崩潰 ,實際使用過程中一不注意就可能掉坑里了。

異常捕獲

async 方法有三種返回類型: void、Task、Task

async void

該方式聲明的方法是無法使用 catch 捕獲異常的,所以以下代碼的 try、catch 并沒什么卵用。

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
private static async void ThrowExceptionAsync()
{
 await Task.Delay(1000);
 throw new Exception("拋個異常玩玩");
}
public static async void CatchAsyncVoidException()
{
 try
 {
  ThrowExceptionAsync();
 }
 catch (Exception ex)
 {
 throw ex;
 }
}

async Task 或 async Task

這兩種方式聲明的方法異常信息會包含Task屬性內(nèi),但前提需要在try里面使用 await 等待。

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
private static async Task ThrowExceptionAsync()
{
 await Task.Delay(1000);
 throw new Exception("拋個異常玩玩");
}
public static async Task CatchAsyncTaskException()
{
 try
 {
   await ThrowExceptionAsync();
 }
 catch (Exception ex)
 {
 throw ex;
 }
}
TaskScheduler.UnobservedTaskException

未捕獲的 Task 異常信息可以通過設(shè)置全局的TaskScheduler.UnobservedTaskException 來記錄錯誤日志,在 Global.asax 增加如下代碼:

?
1
2
3
4
5
6
7
8
9
10
11
12
void Application_Start(object sender, EventArgs e)
{
 // 在應(yīng)用程序啟動時運行的代碼
 TaskScheduler.UnobservedTaskException += TaskScheduler_UnobservedTaskExceptionException;
}
void TaskScheduler_UnobservedTaskExceptionException(object sender, UnobservedTaskExceptionEventArgs e)
{
 if (e.Exception != null)
 {
 // do something
 }
}

同步上下文

異步編程必然是關(guān)于線程的使用,線程有一個同步上下文的概念,個人認(rèn)為線程同步上下文是 async/await 遇到最揪心的問題。在現(xiàn)有項目開發(fā)中我們可能想嘗試使用 async/await,但老代碼都是同步方式,這時如果調(diào)用一個聲明為 async 的方法,死鎖和應(yīng)用程序崩潰的問題一不小心就可能出現(xiàn)。

注意:控制臺程序和.Net Core程序 將不會遇到這個問題,它們不需要同步上下文。

死鎖

?
1
2
3
4
5
6
7
8
9
10
private static async Task XXXAsync()
{
 await Task.Delay(1000);
  // some code
}
public static void Test()
{
 var task = XXXAsync();
 task.Wait();
}

以上代碼很完美的實現(xiàn)了死鎖。 默認(rèn)情況下,當(dāng) Wait() 未完成的 Task 時,會捕獲當(dāng)前線程上下文,在 Task 完成時使用該上下文恢復(fù)方法的執(zhí)行。 當(dāng) async 方法內(nèi)的 await 執(zhí)行完成時,它會嘗試獲取調(diào)用者線程所在的上下文執(zhí)行方法的剩余部分, 但是該上下文已含有一個線程,該線程在等待 async 方法完成。然后它們相互等待對方,然后就沒有然后了,死在那里。

針對死鎖問題的解決方式是增加 ConfigureAwait(false)

?
1
2
// await Task.Delay(1000);
await Task.Delay(1000).ConfigureAwait(false); // 解決死鎖

當(dāng) await 等待完成時,它會嘗試在線程池上下文中執(zhí)行 async 方法的剩余部分,因此就不存在死鎖。

應(yīng)用程序崩潰

測試環(huán)境總發(fā)現(xiàn)IIS應(yīng)用程序池總是崩潰,到底是什么原因?當(dāng)時我們對這個問題也是非常懵逼,代碼看上去并沒什么明顯毛病,試圖欺騙自己應(yīng)該是環(huán)境本身的問題吧,但事實上確實是在代碼中下了毒。通過各種資料查閱和測試,基本可以斷定是同步代碼調(diào)用異步代碼,同步上下文引起的問題。

如果調(diào)用一個 async 方法。如果使用 await 等待,當(dāng)前線程立馬被釋放回線程池,線程的上下文信息會被保存。如果沒有使用 await(async void 的方法,必然沒有辦法使用 await),調(diào)用 async 方法之后,代碼會繼續(xù)往下執(zhí)行,執(zhí)行完成后當(dāng)前線程被釋放回線程池,線程的上下文信息不會被保存。當(dāng) async 中的異步任務(wù)執(zhí)行完成后,會從線程池中獲取一個線程繼續(xù)執(zhí)行剩余代碼,同時會獲取當(dāng)初調(diào)用者所在線程的上下文信息(如果當(dāng)初調(diào)用者所在線程沒有釋放回線程池,上下文信息可以獲取到)。那么問題就來了,如果當(dāng)初調(diào)用者沒有使用 await 并且 所在線程釋放回線程池了,上下文信息因為沒有被保持下來,就獲取不到了,這時候會拋出異常 未將對象引用設(shè)置到對象的實例 ,經(jīng)過測試這個異常信息并不一定每次都會出現(xiàn),原因和線程的釋放有關(guān),調(diào)用者所在線程的上下文信息存在就不會拋出異常。異常錯誤信息如下,這個異常最終會導(dǎo)致應(yīng)用程序集停止。

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
在 System.Web.ThreadContext.AssociateWithCurrentThread(Boolean setImpersonationContext)
  在 System.Web.HttpApplication.OnThreadEnterPrivate(Boolean setImpersonationContext)
  在 System.Web.LegacyAspNetSynchronizationContext.CallCallbackPossiblyUnderLock(SendOrPostCallback callback, Object state)
  在 System.Web.LegacyAspNetSynchronizationContext.CallCallback(SendOrPostCallback callback, Object state)
  在 System.Web.LegacyAspNetSynchronizationContext.Post(SendOrPostCallback callback, Object state)
  在 System.Threading.Tasks.SynchronizationContextAwaitTaskContinuation.PostAction(Object state)
  在 System.Threading.Tasks.AwaitTaskContinuation.RunCallback(ContextCallback callback, Object state, Task& currentTask)
--- 引發(fā)異常的上一位置中堆棧跟蹤的末尾 ---
  在 System.Threading.Tasks.AwaitTaskContinuation.<>c.<ThrowAsyncIfNecessary>b__18_0(Object s)
  在 System.Threading.QueueUserWorkItemCallback.WaitCallback_Context(Object state)
  在 System.Threading.ExecutionContext.RunInternal(ExecutionContext executionContext, ContextCallback callback, Object state, Boolean preserveSyncCtx)
  在 System.Threading.ExecutionContext.Run(ExecutionContext executionContext, ContextCallback callback, Object state, Boolean preserveSyncCtx)
  在 System.Threading.QueueUserWorkItemCallback.System.Threading.IThreadPoolWorkItem.ExecuteWorkItem()
  在 System.Threading.ThreadPoolWorkQueue.Dispatch()
  在 System.Threading._ThreadPoolWaitCallback.PerformWaitCallback()

針對以上異常我們有什么方式可以解決呢?依然是ConfigureAwait(false),在 Task 上加上 ConfigureAwait(false),此設(shè)置代表當(dāng) async 中的異步任務(wù)完成后,不讀取當(dāng)時調(diào)用它的原線程的上下文信息,而是在線程池上下文中執(zhí)行 async 方法的剩余部分。

?
1
2
3
4
5
6
7
public static Task XXXAsync()
{
 await Task.Run(() =>
 {
 // some code
 }).ConfigureAwait(false);
}

總結(jié)

以上所述是小編給大家介紹的ASP.NET 謹(jǐn)用 async/await,希望對大家有所幫助,如果大家有任何疑問請給我留言,小編會及時回復(fù)大家的。在此也非常感謝大家對服務(wù)器之家網(wǎng)站的支持!

原文鏈接:http://beckjin.com/2018/01/07/aspnet-async-await/?utm_source=tuicool&utm_medium=referral

延伸 · 閱讀

精彩推薦
主站蜘蛛池模板: 国产精品久久久久久久久久99 | 精品成人av | 国产片性视频免费播放 | 日韩欧美一区二区精品 | 日韩国产欧美精品 | 精品久久久久久久久久久久久久 | 性做久久久久久久久 | 日韩中文字幕一区 | 免费成人小视频 | 日日操夜| 成人在线小视频 | 国产黄色小视频在线观看 | 久久久久久国产免费 | 久久精品亚洲精品 | 日本久久久 | 综合五月 | 欧美成人精品一区 | 国产精品久久久久免费a∨ 欧美黄色精品 | 欧美日在线| 欧美成人精品一区二区三区 | 欧美一区在线视频 | 能看的av网站 | 精品久久99 | 亚洲 视频 一区 | 免费毛片视频 | 欧美日韩一级二级三级 | 欧美日韩在线免费观看 | 一区二区三区四区视频 | 欧美国产综合 | 精品久久久久久久久久久 | 高清中文字幕av | 成人影院在线观看 | 日韩在线精品强乱中文字幕 | 在线视频三级 | 亚洲欧美日韩另类一区二区 | 亚洲国产中文字幕在线观看 | 久草天堂 | av电影免费| 秋霞电影院午夜伦 | 久久激情五月丁香伊人 | 成人日韩在线观看 |