ASP.NET Core Web 應用程式系列(四)- ASP.NET Core 非同步編程之async await

来源:https://www.cnblogs.com/xyh9039/archive/2019/08/21/11391507.html
-Advertisement-
Play Games

本系列將和大家分享下ASP.NET Core Web 應用程式的一些基礎知識,本章主要分享ASP.NET Core 非同步編程之async await的應用。 ...


PS:非同步編程的本質就是新開任務線程來處理。

約定:非同步的方法名均以Async結尾。

實際上呢,非同步編程就是通過Task.Run()來實現的。

瞭解線程的人都知道,新開一個線程來處理事務這個很常見,但是在以往是沒辦法接收線程裡面返回的值的。所以這時候就該await出場了,await從字面意思不難理解,就是等待的意思。

執行await的方法必須是async修飾的,並且是Task的類型。 非同步執行後,返回的信息存儲在result屬性中。但並非主進程就會卡在await行的代碼上,執行到await方法之後主線程繼續往下執行,無需等待新的線程執行完再繼續。只有當需要用到新線程返回的result結果時,此時主進程才會等待新線程執行完並返回內容。也就是說,若無需用到新線程返回的結果,那麼主進程不會等待。

async和await呢,返回類型就3種:void、Task、Task<TResult>。

1、void

如果在觸發後,你懶得管,請使用 void。

void返回類型主要用在事件處理程式中,一種稱為“fire and forget”(觸發並忘記)的活動的方法。除了它之外,我們都應該儘可能是用Task,作為我們非同步方法的返回值。

返回void,意味著不能await該非同步方法,即可能出現線程阻塞,並且也無法獲取exception拋出的異常,通常這些異常會導致我們的程式失敗,如果你使用的是Task和Task<TResult>,catch到的異常會包裝在屬性裡面,調用方法就可以從中獲取異常信息,並選擇正確的處理方式。

2、Task

你如果只是想知道執行的狀態,而不需要一個具體的返回結果時,請使用Task。
與void對比呢,Task可以使用await進行等待新線程執行完畢。而void不需要等待。

3、Task<TResult> 

當你添加async關鍵字後,需要返回一個將用於後續操作的對象,請使用Task<TResult>。

主要有兩種方式獲取結果值,一個是使用Result屬性,一個是使用await。他們的區別在於:如果你使用的是Result,它帶有阻塞性,即在任務完成之前進行訪問讀取它,當前處於活動狀態的線程都會出現阻塞的情形,一直到結果值可用。所以,在絕大多數情況下,除非你有絕對的理由告訴自己,否則都應該使用await,而不是屬性Result來讀取結果值。

 

接下來我們來看個例子,在上一章的基礎上我們添加非同步的方法。

首先是倉儲層介面:

using System;
using System.Collections.Generic;
using System.Text;
using System.Threading.Tasks;

using TianYa.DotNetShare.Model;

namespace TianYa.DotNetShare.Repository
{
    /// <summary>
    /// 學生類倉儲層介面
    /// </summary>
    public interface IStudentRepository
    {
        /// <summary>
        /// 根據學號獲取學生信息
        /// </summary>
        /// <param name="stuNo">學號</param>
        /// <returns>學生信息</returns>
        Student GetStuInfo(string stuNo);

        /// <summary>
        /// 根據學號獲取學生信息(非同步)
        /// </summary>
        /// <param name="stuNo">學號</param>
        /// <returns>學生信息</returns>
        Task<Student> GetStuInfoAsync(string stuNo);
    }
}

接著是倉儲層實現:

using System;
using System.Collections.Generic;
using System.Text;
using System.Threading.Tasks;

using TianYa.DotNetShare.Model;

namespace TianYa.DotNetShare.Repository.Impl
{
    /// <summary>
    /// 學生類倉儲層
    /// </summary>
    public class StudentRepository : IStudentRepository
    {
        /// <summary>
        /// 根據學號獲取學生信息
        /// </summary>
        /// <param name="stuNo">學號</param>
        /// <returns>學生信息</returns>
        public Student GetStuInfo(string stuNo)
        {
            //數據訪問邏輯,此處為了演示就簡單些
            var student = new Student();
            switch (stuNo)
            {
                case "10000":
                    student = new Student() { StuNo = "10000", Name = "張三", Sex = "", Age = 20 };
                    break;
                case "10001":
                    student = new Student() { StuNo = "10001", Name = "錢七七", Sex = "", Age = 18 };
                    break;
                case "10002":
                    student = new Student() { StuNo = "10002", Name = "李四", Sex = "", Age = 21 };
                    break;
                default:
                    student = new Student() { StuNo = "10003", Name = "王五", Sex = "", Age = 25 };
                    break;
            }

            return student;
        }

        /// <summary>
        /// 根據學號獲取學生信息(非同步)
        /// </summary>
        /// <param name="stuNo">學號</param>
        /// <returns>學生信息</returns>
        public virtual async Task<Student> GetStuInfoAsync(string stuNo)
        {
            return await Task.Run(() => this.GetStuInfo(stuNo));
        }
    }
}

然後是服務層介面:

using System;
using System.Collections.Generic;
using System.Text;
using System.Threading.Tasks;

using TianYa.DotNetShare.Model;

namespace TianYa.DotNetShare.Service
{
    /// <summary>
    /// 學生類服務層介面
    /// </summary>
    public interface IStudentService
    {
        /// <summary>
        /// 根據學號獲取學生信息
        /// </summary>
        /// <param name="stuNo">學號</param>
        /// <returns>學生信息</returns>
        Student GetStuInfo(string stuNo);

        /// <summary>
        /// 根據學號獲取學生信息(非同步)
        /// </summary>
        /// <param name="stuNo">學號</param>
        /// <returns>學生信息</returns>
        Task<Student> GetStuInfoAsync(string stuNo);
    }
}

再接著是服務層實現:

using System;
using System.Collections.Generic;
using System.Text;
using System.Threading.Tasks;

using TianYa.DotNetShare.Model;
using TianYa.DotNetShare.Repository;

namespace TianYa.DotNetShare.Service.Impl
{
    /// <summary>
    /// 學生類服務層
    /// </summary>
    public class StudentService : IStudentService
    {
        /// <summary>
        /// 定義倉儲層學生抽象類對象
        /// </summary>
        protected IStudentRepository StuRepository;

        /// <summary>
        /// 空構造函數
        /// </summary>
        public StudentService() { }

        /// <summary>
        /// 構造函數
        /// </summary>
        /// <param name="stuRepository">倉儲層學生抽象類對象</param>
        public StudentService(IStudentRepository stuRepository)
        {
            this.StuRepository = stuRepository;
        }

        /// <summary>
        /// 根據學號獲取學生信息
        /// </summary>
        /// <param name="stuNo">學號</param>
        /// <returns>學生信息</returns>
        public Student GetStuInfo(string stuNo)
        {
            var stu = StuRepository.GetStuInfo(stuNo);
            return stu;
        }

        /// <summary>
        /// 根據學號獲取學生信息(非同步)
        /// </summary>
        /// <param name="stuNo">學號</param>
        /// <returns>學生信息</returns>
        public virtual async Task<Student> GetStuInfoAsync(string stuNo)
        {
            return await StuRepository.GetStuInfoAsync(stuNo);
        }
    }
}

最後進行控制器中的調用:

using System;
using System.Collections.Generic;
using System.Diagnostics;
using System.Linq;
using System.Threading.Tasks;
using Microsoft.AspNetCore.Mvc;
using TianYa.DotNetShare.CoreAutofacDemo.Models;

using TianYa.DotNetShare.Service;
using TianYa.DotNetShare.Repository;
using TianYa.DotNetShare.Repository.Impl;

namespace TianYa.DotNetShare.CoreAutofacDemo.Controllers
{
    public class HomeController : Controller
    {
        /// <summary>
        /// 定義倉儲層學生抽象類對象
        /// </summary>
        private IStudentRepository _stuRepository;

        /// <summary>
        /// 定義服務層學生抽象類對象
        /// </summary>
        private IStudentService _stuService;

        /// <summary>
        /// 定義服務層學生抽象類對象
        /// 屬性註入:訪問修飾符必須為public,否則會註入失敗。
        /// </summary>
        public IStudentService StuService { get; set; }

        /// <summary>
        /// 定義倉儲層學生實現類對象
        /// 屬性註入:訪問修飾符必須為public,否則會註入失敗。
        /// </summary>
        public StudentRepository StuRepository { get; set; }

        /// <summary>
        /// 通過構造函數進行註入
        /// 註意:參數是抽象類,而非實現類,因為已經在Startup.cs中將實現類映射給了抽象類
        /// </summary>
        /// <param name="stuRepository">倉儲層學生抽象類對象</param>
        /// <param name="stuService">服務層學生抽象類對象</param>
        public HomeController(IStudentRepository stuRepository, IStudentService stuService)
        {
            this._stuRepository = stuRepository;
            this._stuService = stuService;
        }

        public IActionResult Index()
        {
            var stu1 = StuRepository.GetStuInfo("10000");
            var stu2 = StuService.GetStuInfo("10001");
            var stu3 = _stuService.GetStuInfo("10002");
            var stu4 = _stuRepository.GetStuInfo("1003");
            string msg = $"學號:10000,姓名:{stu1.Name},性別:{stu1.Sex},年齡:{stu1.Age}<br />";
            msg += $"學號:10001,姓名:{stu2.Name},性別:{stu2.Sex},年齡:{stu2.Age}<br/>";
            msg += $"學號:10002,姓名:{stu3.Name},性別:{stu3.Sex},年齡:{stu3.Age}<br/>";
            msg += $"學號:10003,姓名:{stu4.Name},性別:{stu4.Sex},年齡:{stu4.Age}<br/>";

            return Content(msg, "text/html", System.Text.Encoding.UTF8);
        }

        public async Task<IActionResult> Privacy()
        {
            var stu = await _stuService.GetStuInfoAsync("10000");
            return Json(stu);
        }

        [ResponseCache(Duration = 0, Location = ResponseCacheLocation.None, NoStore = true)]
        public IActionResult Error()
        {
            return View(new ErrorViewModel { RequestId = Activity.Current?.Id ?? HttpContext.TraceIdentifier });
        }
    }
}

至此完成處理,我們來訪問一下/home/privacy,看看是否正常

可以看出是正常的

下麵我們演示一下什麼時候需要用到result屬性:

//用了await則不需要Result屬性
public async Task<IActionResult> Privacy()
{
    var stu = await _stuService.GetStuInfoAsync("10000");
    return Json(stu);
}
//沒有用await則需要Result屬性
public async Task<IActionResult> Privacy()
{
    var stu = _stuService.GetStuInfoAsync("10000").Result;
    return Json(stu);
}

至此我們的非同步編程就講解完了。

總結:

1、儘量優先使用Task<TResult>和Task作為非同步方法的返回類型。

2、如果用了await則方法必須使用async來修飾,並且是Task的類型。

demo源碼:

鏈接:https://pan.baidu.com/s/1Wb0Mebm-nh9YFOaYNLwO-g 
提取碼:1ayv

 

參考博文:https://www.cnblogs.com/fei686868/p/9637310.html

版權聲明:如有雷同純屬巧合,如有侵權請及時聯繫本人修改,謝謝!!!


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 昨天花了一下午寫了一個小爬蟲,用來分析自己的粉絲數據。這個真好玩!今天幫了群里好多大V也爬了他們的數據。運行速度:每分鐘5千粉絲以上。暫時先寫成這樣,這兩天要準備補考,沒有時間繼續玩這個。 下次要改進的地方:1、多線程 2、scrapy 3、深度數據 4、分散式爬蟲 希望實現的功能: + 1、地區、 ...
  • 綁定事件: 解綁事件: 點擊事件:click 滑鼠:mousedown、mouseup、mousemove、mouseover、mouseout、mouseenter、mouseleave 對於input框:focus、blur、input input能夠實時檢測 textarea,input:te ...
  • unittest中的測試斷言分兩天總結,hhh其實內容不多,就是懶~ 斷言的作用是什麼? 答:設置測試斷言以後,能幫助我們判斷測試用例執行結果。 我們先看下unittest支持的斷言有哪些: 對上面的斷言語法有個大概的瞭解後,我們使用一下看看代碼: 一: 註意:相等,必須是內容和類型都完全相等哦,比 ...
  • String類常用的構造方法 1、String(byte[] bytes) 通過使用平臺的預設字元集解碼指定的 byte 數組,構造一個新的 String。 2、String(char[] value) 分配一個新的 String,使其表示字元數組參數中當前包含的字元序列。 3、String(cha ...
  • 之前操作了一個IDC網站,不到1個月的時間把網站的權重從0做到了1,本來想寫篇文章分享相關的操作經驗。後來因為網站整體規劃的原因,IDC網站需要關閉一段時間做備案的更新,排名肯定就會掉了,然後怕大家看到我後面網站的數據不是我分享說的樣子,說我騙人,就沒寫那次的分享經驗。 今天無意間通過站長工具查詢新 ...
  • 一、排序演算法 1、冒泡排序(Bubble Sort) 定義:是一種簡單的排序演算法。它重覆地遍歷要排序的數列,一次比較兩個元素,如果他們的順序錯誤就把他們交換過來。遍曆數列的工作是重覆地進行直到沒有再需要交換,也就是說該數列已經排序完成。這個演算法的名字由來是因為越小的元素會經由交換慢慢“浮”到數列的頂 ...
  • 本人.net 程式猿一枚,平常最大的愛好就是coding ,尤其是對c#情有獨衷,數年前,因為某個同事的一句話“大公司都有自己的代碼框架,甚至是代碼工具”,因為當時在一個小公司就職,而我也只是一個小小的團隊帶頭人,心裡就莫名有了一個想法,做一款代碼工具,一款很有自我特色,從公司項目開發,到項目維護的 ...
  • 看到一篇公眾號文章《2020年什麼編程語言最受歡迎,待遇最高?》,其中對C#的描述如下:點擊閱讀原文,看到這是一篇翻譯文章:https://codinginfinite.com/top-programming-languages-2020-stats-surveys/這篇文章里列出的那些語言之中,明... ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...