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
  • 1. 說明 /* Performs operations on System.String instances that contain file or directory path information. These operations are performed in a cross-pla ...
  • 視頻地址:【WebApi+Vue3從0到1搭建《許可權管理系統》系列視頻:搭建JWT系統鑒權-嗶哩嗶哩】 https://b23.tv/R6cOcDO qq群:801913255 一、在appsettings.json中設置鑒權屬性 /*jwt鑒權*/ "JwtSetting": { "Issuer" ...
  • 引言 集成測試可在包含應用支持基礎結構(如資料庫、文件系統和網路)的級別上確保應用組件功能正常。 ASP.NET Core 通過將單元測試框架與測試 Web 主機和記憶體中測試伺服器結合使用來支持集成測試。 簡介 集成測試與單元測試相比,能夠在更廣泛的級別上評估應用的組件,確認多個組件一起工作以生成預 ...
  • 在.NET Emit編程中,我們探討了運算操作指令的重要性和應用。這些指令包括各種數學運算、位操作和比較操作,能夠在動態生成的代碼中實現對數據的處理和操作。通過這些指令,開發人員可以靈活地進行算術運算、邏輯運算和比較操作,從而實現各種複雜的演算法和邏輯......本篇之後,將進入第七部分:實戰項目 ...
  • 前言 多表頭表格是一個常見的業務需求,然而WPF中卻沒有預設實現這個功能,得益於WPF強大的控制項模板設計,我們可以通過修改控制項模板的方式自己實現它。 一、需求分析 下圖為一個典型的統計表格,統計1-12月的數據。 此時我們有一個需求,需要將月份按季度劃分,以便能夠直觀地看到季度統計數據,以下為該需求 ...
  • 如何將 ASP.NET Core MVC 項目的視圖分離到另一個項目 在當下這個年代 SPA 已是主流,人們早已忘記了 MVC 以及 Razor 的故事。但是在某些場景下 SSR 還是有意想不到效果。比如某些靜態頁面,比如追求首屏載入速度的時候。最近在項目中回歸傳統效果還是不錯。 有的時候我們希望將 ...
  • System.AggregateException: 發生一個或多個錯誤。 > Microsoft.WebTools.Shared.Exceptions.WebToolsException: 生成失敗。檢查輸出視窗瞭解更多詳細信息。 內部異常堆棧跟蹤的結尾 > (內部異常 #0) Microsoft ...
  • 引言 在上一章節我們實戰了在Asp.Net Core中的項目實戰,這一章節講解一下如何測試Asp.Net Core的中間件。 TestServer 還記得我們在集成測試中提供的TestServer嗎? TestServer 是由 Microsoft.AspNetCore.TestHost 包提供的。 ...
  • 在發現結果為真的WHEN子句時,CASE表達式的真假值判斷會終止,剩餘的WHEN子句會被忽略: CASE WHEN col_1 IN ('a', 'b') THEN '第一' WHEN col_1 IN ('a') THEN '第二' ELSE '其他' END 註意: 統一各分支返回的數據類型. ...
  • 在C#編程世界中,語法的精妙之處往往體現在那些看似微小卻極具影響力的符號與結構之中。其中,“_ =” 這一組合突然出現還真不知道什麼意思。本文將深入剖析“_ =” 的含義、工作原理及其在實際編程中的廣泛應用,揭示其作為C#語法奇兵的重要角色。 一、下劃線 _:神秘的棄元符號 下劃線 _ 在C#中並非 ...