【转】2.3async中必须始终返回Task(@Ron.liang)
Asp.Net Core 輕松學(xué)-經(jīng)常使用異步的你,可能需要看看這個(gè)文章
目錄
- 前言
- 1. 異常的發(fā)生來(lái)得太突然
- 2. 問(wèn)題所在
- 3. 問(wèn)題的解決方案
?
前言
事情的起因是由于一段簡(jiǎn)單的數(shù)據(jù)庫(kù)連接代碼引起,這段代碼從語(yǔ)法上看,是沒(méi)有任何問(wèn)題;但是就是莫名其妙的報(bào)錯(cuò)了,這段代碼極其簡(jiǎn)單,就是打開(kāi)數(shù)據(jù)庫(kù)連接,讀取一條記錄,然后立即更新到數(shù)據(jù)庫(kù)中。但是,慘痛的事實(shí)證明,老司機(jī)也是會(huì)翻車的。
1. 異常的發(fā)生來(lái)得太突然
1.1 引起不舒適的代碼片段
[HttpPut]public async void Put([FromBody] TopicViewModel model){var topic = this.context.Topics.Where(f => f.Id == model.Id).FirstOrDefault();topic.Content = model.Content;this.context.Update(topic);var affrows = await this.context.SaveChangesAsync();}這是一段不太標(biāo)準(zhǔn)的異步接口,可能你也這么寫過(guò), 從結(jié)構(gòu)和語(yǔ)法上看,這段代碼沒(méi)有任何問(wèn)題,而且正常情況下,它還能執(zhí)行成功
1.2 報(bào)錯(cuò)信息
從報(bào)錯(cuò)信息中可以看出,數(shù)據(jù)庫(kù)上下文對(duì)象被銷毀了,是在什么時(shí)候銷毀的呢,通過(guò)跟蹤程序,了解到,是在 this.context.Update(topic); ,調(diào)用 Update 后執(zhí)行了 DbContext.Dispose(),為了證明這點(diǎn),我們重寫 DbContext.Dispose() 方法,并簡(jiǎn)單的輸出一句話
1.3 重寫 DbContext.Dispose()
public class ForumContext : DbContext{public ForumContext(DbContextOptions<ForumContext> options) : base(options){}public DbSet<Topic> Topics { get; set; }public DbSet<Post> Posts { get; set; }public override void Dispose(){base.Dispose();Console.WriteLine("Dispose");}}1.4 再次執(zhí)行程序,查看結(jié)果
通過(guò)輸出結(jié)果紅色方框處可以看到,確實(shí)是在執(zhí)行了 Update 以后執(zhí)行了 Dispose 方法,關(guān)于這點(diǎn),如果我們使用了同步方法,先 Update 再 SaveChanges ,這是沒(méi)有任何問(wèn)題的,理論上說(shuō),EFCore 中啟用了 AutoDetectChangesEnabled,我們?cè)谏厦娴拇a中其實(shí)無(wú)需調(diào)用 Update,直接 SaveChangesAsync 即可,也不會(huì)拋出異常,同理,如果是在同步方法中,先執(zhí)行 Update 再 SaveChanges ,也是沒(méi)有任何問(wèn)題的
1.5 同步 SaveChanges
[HttpPut]public void Put([FromBody] TopicViewModel model){var topic = this.context.Topics.Where(f => f.Id == model.Id).FirstOrDefault();topic.Content = model.Content;this.context.Update(topic);Console.WriteLine("Updated");var affrows = this.context.SaveChanges();Console.WriteLine("affrows:{0}", affrows);}- 輸出結(jié)果
從輸出結(jié)果可知,先執(zhí)行了 Update,然后執(zhí)行了 SaveChanges 輸出 affrows,最后執(zhí)行了 Dispose 方法
2. 問(wèn)題所在
那到底是什么問(wèn)題引起了程序執(zhí)行的不確定性呢,答案就是 async/await,我們先來(lái)嘗試改進(jìn)一下最初的代碼
2.1 改進(jìn)后的代碼
[HttpPut]public async Task Put([FromBody] TopicViewModel model){var topic = this.context.Topics.Where(f => f.Id == model.Id).FirstOrDefault();topic.Content = model.Content;this.context.Update(topic);Console.WriteLine("Updated");var affrows = await this.context.SaveChangesAsync();Console.WriteLine("affrows:{0}", affrows);}細(xì)心的你已經(jīng)發(fā)現(xiàn),這段代碼和 1.1 之中的沒(méi)有太多的不同,無(wú)非是增加了一些跟蹤信息,其中,最關(guān)鍵的是:增加了返回值為:Task ,替換了 void
2.2 再次執(zhí)行修正的程序
輸出結(jié)果和 1.5 中的同步方法完全相同,至此,問(wèn)題解決
3. 問(wèn)題的解決方案
3.1 問(wèn)題分析
為什么會(huì)發(fā)生這種問(wèn)題呢,原因就是因?yàn)槭褂昧水惒椒椒?async/await 時(shí),當(dāng)沒(méi)有值需要返回時(shí),使用了 void 造成的,正確的做法是如果沒(méi)有返回值,則返回 Task,如果有返回值,則使用 Task?;當(dāng)一個(gè)異步方法內(nèi)部沒(méi)有返回 Task 的時(shí)候,基于任務(wù)的異步模式(TAP)并不知道異步任務(wù)的狀態(tài),當(dāng) this.context.Update 執(zhí)行完成后,發(fā)現(xiàn)掛載在內(nèi)存中的連接已經(jīng)沒(méi)有使用,就執(zhí)行了回收;實(shí)際上,此時(shí)程序還沒(méi)有執(zhí)行完成,但是 TAP 并不知道,所以它不會(huì)去阻止這個(gè)回收的過(guò)程(使用標(biāo)記),所以 async/await 應(yīng)該成對(duì)出現(xiàn),并且應(yīng)該始終返回 Task 或者 Task,以確保 TAP 能夠?qū)⑸舷挛倪M(jìn)行正確的掛載,否則,當(dāng)異常發(fā)生時(shí),TAP 無(wú)非將異常信息掛載到相應(yīng)的 Task 上,亦無(wú)法跟蹤其執(zhí)行狀態(tài)等信息
3.2 解決方案
請(qǐng)牢記下面的鐵律
- 3.2.1 在 EFCore 中,應(yīng)當(dāng)始終發(fā)揮 AutoDetectChangesEnabled 的特性,不要再更新實(shí)體的時(shí)候去調(diào)用 Update 方法
- 3.2.2 使用 async/await 修飾方法時(shí),應(yīng)該始終返回 Task 或者 Task
- 適當(dāng)?shù)氖褂猛椒椒?#xff0c;可避免異步踩坑
演示代碼下載
https://github.com/lianggx/EasyAspNetCoreDemo/tree/master/Ron.TaskThird
總結(jié)
以上是生活随笔為你收集整理的【转】2.3async中必须始终返回Task(@Ron.liang)的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問(wèn)題。
- 上一篇: SharePoint无代码工作流设计开发
- 下一篇: 网申信用卡多久有结果