net core WebApi——使用xUnits來實現單元測試

来源:https://www.cnblogs.com/AprilBlank/archive/2019/11/06/11803159.html
-Advertisement-
Play Games

[toc] 前言 從開始敲代碼到現在,不停地都是在喊著記得做測試,記得自測,測試人員打回來扣你money之類的,剛開始因為心疼錢(當然還是為了代碼質量),就老老實實自己寫完自己跑一遍,沒有流程沒有規劃沒有測試文檔,就是自己整理一組數據跑一遍,最後依然還是讓測試人員老老實實把一大堆測試問題扔給你。 單 ...


目錄

前言

從開始敲代碼到現在,不停地都是在喊著記得做測試,記得自測,測試人員打回來扣你money之類的,剛開始因為心疼錢(當然還是為了代碼質量),就老老實實自己寫完自己跑一遍,沒有流程沒有規劃沒有測試文檔,就是自己整理一組數據跑一遍,最後依然還是讓測試人員老老實實把一大堆測試問題扔給你。

單元測試

首先,還是來聊聊為啥要搞測試吧。

  1. 測試有助於代碼整體健壯性,覆蓋測試、壓力測試都是為了全方位多角度更快更好為用戶服務。
  2. 測試有助於提高程式猿的積極性以及引起自身的重視,畢竟一個坑栽一遍就夠了,兩次也能容忍,再三再四再五怕是要被搞,同時這也是自我提高的一種手段吧。
  3. 軟體開發流程收尾的工作就是測試,繞不過,畢竟驗收才是最終目標,達到效果才能獲得應有的。

好了,聊完這些,當然我也不是專業測試人員,肯定不會給個測試文檔模板,喏,照著這個規範起來,我主要是要鼓搗下我之前一直想試試的單元測試,這個自動化測試的手段之一,一直想試試但是一直都放著。

MSTestNUnitxUint這三個中讓我稍微猶豫了下,不過三七二十八管他呢,隨便來個吧,就選了xUnit,當然MSTest是官方的,支持度應該高點兒,但是這不是我們該猶豫抉擇的地方。

xUnit

首先,我們新建一個項目April.Test
新建

Fact

新建之後,我們看到有個預設的 [Fact]

這個就是測試的標準格式了,如果我們不需要測試數據的話,就是在這個標簽下的方法做斷言,簡單舉個例子吧。

        [Fact]
        public void TestEqual()
        {
            int a = 10, b = 20;
            Assert.Equal(30, Add(a, b));
        }

        private int Add(int a, int b)
        {
            return a + b;
        }

敲下Assert之後,發現斷言好多方法,這裡也就不一一說明瞭,感覺方法名起的都挺明白的,這裡就不過多描述了,代碼中也有些測試示例,地址在文末給出。

Theory

我們在上面的代碼可以看到,所有的數據都是自己定好的,如果我想自己做參數傳入測試怎麼搞呢,這時候就要用到 [Theory] 這個標簽以及 [InlineData] 了,標簽是為了更好的區分方法的類型(個人理解),來看下這種參數傳遞的測試怎麼搞吧。

        [Theory]
        [InlineData(new object[] { 1, 2, 3, 4 },1)]
        [InlineData(new object[] { "t", "e", "s", "t" }, "t")]
        public void TestContains(object[] objs,object obj)
        {
            Assert.Contains(obj, objs);
        }

當然我們也可以使用自定義數組來做測試數據源,這裡我起初以為可以傳任意類型參數,但是MemberData只支持object[]。

        [Theory]
        [MemberData(nameof(tempDatas))]
        public void TestData(int a, int b)
        {
            int result = a + b;
            Assert.True(result == Add(a, b));
        }
        
        public static IEnumerable<object[]> tempDatas
        {
            get
            {
                yield return new object[] { 1, 2 };
                yield return new object[] { 5, 7 };
                yield return new object[] { 12, 12 };
            }
        }

控制器

在之前鼓搗單元測試的時候,我一直想一個問題,如果只是這種操作的話,那測試有何意義,但是後來發現,原來單元測試比我以為能做的多得多,所以,學習是個不停的行程,走的多了,風景也就多了。

這裡可以引入一個流程,在我們測試自己的工程的時候,我們需要三步來做完單個的測試。

  • Arrange(準備工作)
  • Act(實現方法)
  • Assert(斷言結果)

在測試之前,我們需要在當前工程引入一個Moq,至於為什麼用這個呢,就跟vue一樣我們需要模擬(雖說mock跟Moq還不是那麼類似),總不可能我們單元測試引入了orm來實際操作資料庫吧(當然測試庫這個也是可行的),所以我們需要模擬介面的實現類及方法。

引入

引入完,我們將April.WebApi引入到當前工程,之後我們創建一個Values的介面測試類ValuesControllerTest,然後測試方法如下:

        [Fact]
        public void TestGet()
        {
            // Arrange
            var mockRepo = new Mock<IStudentService>();
            var controller = new ValuesController(mockRepo.Object);
            // Act
            var result = controller.Get();
            // Assert
            Assert.Equal(new string[] { "value1", "" }, result.Value);
        }

這裡註意下,對應介面的方法我是都註釋的,最終只有返回了字元串數組,因為對應介面中的方法都需要初始化,那麼我們來測試下獲取數據的介面,看下這個Moq的用法吧,首先我們需要初始化Mock來實現對應介面,然後我們通過實現對應方法來模擬獲取數據。

        [Theory]
        [InlineData(1)]
        public void TestGetByID(int id)
        {
            var mockRepo = new Mock<IStudentService>();
            mockRepo.Setup(repo => repo.GetList(s => s.ID == 38).ToList())
                .Returns(GetList());

            var controller = new ValuesController(mockRepo.Object);

            var result = controller.Get(id);

            Assert.NotNull(result);
            Assert.Contains("大洛陽", result.Value);
        }

        private List<StudentEntity> GetList()
        {
            List<StudentEntity> entities = new List<StudentEntity>();

            entities.Add(new StudentEntity()
            {
                ID = 1,
                Name = "小明",
                Number = "123456",
                Age = 19,
                Sex = 1,
                Address = "大洛陽"
            });
            entities.Add(new StudentEntity()
            {
                ID = 2,
                Name = "小紅",
                Number = "456789",
                Age = 18,
                Sex = 0,
                Address = "大洛陽"
            });

            return entities;
        }

寫到這我發現,我在引入WebApi的時候,好像已經把幾個工程都已經引入進來,這個不知道合適不合適。

小結

寫到這裡,基本上單元測試這塊兒也簡單了走了一遍,至於具體在業務中如何實現,還是想著需要結合個小項目來實踐下,東西走通個demo只能說明可行,走通不同體量的工程才能說明可用,包括後續的集成測試,壓力測試,自動化測試也會一點點兒的開始鼓搗,路漫漫啊。

附錄

代碼地址:April.WebApi


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

-Advertisement-
Play Games
更多相關文章
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...