欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

淺談.Net Core后端單元測試的實現(xiàn)

 更新時間:2021年03月09日 10:08:55   作者:寶樹吶  
這篇文章主要介紹了淺談.Net Core后端單元測試的實現(xiàn),文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧

1. 前言

單元測試一直都是"好處大家都知道很多,但是因為種種原因沒有實施起來"的一個老大難問題。具體是否應該落地單元測試,以及落地的程度, 每個項目都有自己的情況。

本篇為個人認為"如何更好地寫單元測試", 即更加 偏向實踐向 中夾雜一些理論的分享。

下列示例的單元測試框架為 xUnit , Mock庫為 Moq

2. 為什么需要單元測試

優(yōu)點有很多, 這里提兩點我個人認為的很明顯的好處

2.1 防止回歸

通常在進行新功能/模塊的開發(fā)或者是重構的時候,測試會進行回歸測試原有的已存在的功能,以驗證以前實現(xiàn)的功能是否仍能按預期運行。

使用單元測試,可在每次生成后,甚至在更改一行代碼后重新運行整套測試, 從而可以很大程度減少回歸缺陷。

2.2 減少代碼耦合

當代碼緊密耦合或者一個方法過長的時候,編寫單元測試會變得很困難。當不去做單元測試的時候,可能代碼的耦合不會給人感覺那么明顯。為代碼編寫測試會自然地解耦代碼,變相提高代碼質量和可維護性。

3. 基本原則和規(guī)范

 3.1 3A原則

3A分別是"arrange、act、assert", 分別代表一個合格的單元測試方法的三個階段

  • 事先的準備
  • 測試方法的實際調用
  • 針對返回值的斷言

一個單元測試方法可讀性是編寫測試時最重要的方面之一。 在測試中分離這些操作會明確地突出顯示調用代碼所需的依賴項、調用代碼的方式以及嘗試斷言的內容.

所以在進行單元測試的編寫的時候, 請使用注釋標記出3A的各個階段的, 如下示例

[Fact]
public async Task VisitDataCompressExport_ShouldReturnEmptyResult_WhenFileTokenDoesNotExist()
{
  // arrange
  var mockFiletokenStore = new Mock<IFileTokenStore>();
  mockFiletokenStore
    .Setup(it => it.Get(It.IsAny<string>()))
    .Returns(string.Empty);

  var controller = new StatController(
    mockFiletokenStore.Object,
    null);

  // act
  var actual = await controller.VisitDataCompressExport("faketoken");

  // assert
  Assert.IsType<EmptyResult>(actual);
}

3.2 盡量避免直接測試私有方法

盡管私有方法可以通過反射進行直接測試,但是在大多數(shù)情況下,不需要直接測試私有的private方法, 而是通過測試公共public方法來驗證私有的private方法。

可以這樣認為:private方法永遠不會孤立存在。更應該關心的是調用private方法的public方法的最終結果。

3.3 重構原則

如果一個類/方法,有很多的外部依賴,造成單元測試的編寫困難。那么應該考慮當前的設計和依賴項是否合理。是否有部分可以存在解耦的可能性。選擇性重構原有的方法,而不是硬著頭皮寫下去.

3.4 避免多個斷言

如果一個測試方法存在多個斷言,可能會出現(xiàn)某一個或幾個斷言失敗導致整個方法失敗。這樣不能從根本上知道是了解測試失敗的原因。

所以一般有兩種解決方案

  • 拆分成多個測試方法
  • 使用參數(shù)化測試, 如下示例
[Theory]
[InlineData(null)]
[InlineData("a")]
public void Add_InputNullOrAlphabetic_ThrowsArgumentException(string input)
{
  // arrange
  var stringCalculator = new StringCalculator();

  // act
  Action actual = () => stringCalculator.Add(input);

  // assert
  Assert.Throws<ArgumentException>(actual);
}

當然如果是對對象進行斷言, 可能會對對象的多個屬性都有斷言。此為例外。

3.5 文件和方法命名規(guī)范 文件名規(guī)范

一般有兩種。比如針對 UserController 下方法的單元測試應該統(tǒng)一放在 UserControllerTest 或者 UserController_Test

單元測試方法名

單元測試的方法名應該具有可讀性,讓整個測試方法在不需要注釋說明的情況下可以被讀懂。格式應該類似遵守如下

<被測試方法全名>_<期望的結果>_<給予的條件>

// 例子
[Fact]
public void Add_InputNullOrAlphabetic_ThrowsArgumentException()
{
 ...
}

4. 常用類庫介紹

4.1 xUnit/MsTest/NUnit

編寫.Net Core的單元測試繞不過要選擇一個單元測試的框架, 三大單元測試框架中

  • MsTest是微軟官方出品的一個測試框架
  • NUnit沒用過
  • xUnit是.Net Foundation下的一個開源項目,并且被dotnet github上很多倉庫(包括runtime)使用的單元測試框架

三大測試框架發(fā)展至今已是大差不差, 很多時候選擇只是靠個人的喜好。

個人偏好 xUnit 簡潔的斷言

// xUnit
Assert.True()
Assert.Equal()

// MsTest
Assert.IsTrue()
Assert.AreEqual()

客觀地功能性地分析三大框架地差異可以參考如下

https://anarsolutions.com/automated-unit-testing-tools-comparison

4.2 Moq

官方倉庫

https://github.com/moq/moq4

Moq是一個非常流行的模擬庫, 只要有一個接口它就可以動態(tài)生成一個對象, 底層使用的是Castle的動態(tài)代理功能.

基本用法

在實際使用中可能會有如下場景

public class UserController
{
  private readonly IUserService _userService;
  
  public UserController(IUserService userService)
  {
    _userService = userService;
  }
  
  [HttpGet("{id}")]
  public IActionResult GetUser(int id)
  {
    var user = _userService.GetUser(id);
    
    if (user == null)
    {
      return NotFound();
    }
    else
    {
      ...
    }
  }
}

在進行單元測試的時候, 可以使用 Moq_userService.GetUser 進行模擬返回值

[Fact]
public void GetUser_ShouldReturnNotFound_WhenCannotFoundUser()
{
  // arrange
  // 新建一個IUserService的mock對象
  var mockUserService = new Mock<IUserService>();
  // 使用moq對IUserService的GetUs方法進行mock: 當入?yún)?33時返回null
  mockUserService
   .Setup(it => it.GetUser(233))
   .Return((User)null);
  var controller = new UserController(mockUserService.Object);
  
  // act
  var actual = controller.GetUser(233) as NotFoundResult;
  
  // assert
  // 驗證調用過userService的GetUser方法一次,且入?yún)?33
  mockUserService.Verify(it => it.GetUser(233), Times.AtMostOnce());
}

4.3 AutoFixture

官方倉庫

https://github.com/AutoFixture/AutoFixture

AutoFixture是一個假數(shù)據(jù)填充庫,旨在最小化3A中的 arrange 階段,使開發(fā)人員更容易創(chuàng)建包含測試數(shù)據(jù)的對象,從而可以更專注與測試用例的設計本身。

基本用法

直接使用如下的方式創(chuàng)建強類型的假數(shù)據(jù)

[Fact]
public void IntroductoryTest()
{
  // arrange
  Fixture fixture = new Fixture();

  int expectedNumber = fixture.Create<int>();
  MyClass sut = fixture.Create<MyClass>();
  
  // act
  int result = sut.Echo(expectedNumber);
  
  // assert
  Assert.Equal(expectedNumber, result);
}

上述示例也可以和測試框架本身結合,比如xUnit

[Theory, AutoData]
public void IntroductoryTest(
  int expectedNumber, MyClass sut)
{
  // act
  int result = sut.Echo(expectedNumber);
  
  // assert
  Assert.Equal(expectedNumber, result);
}

5. 實踐中結合Visual Studio的使用

Visual Studio提供了完備的單元測試的支持,包括運行. 編寫. 調試單元測試。以及查看單元測試覆蓋率等。

5.1 如何在Visual Studio中運行單元測試

5.2 如何在Visual Studio中查看單元測試覆蓋率

如下功能需要Visual Studio 2019 Enterprise版本,社區(qū)版不帶這個功能。

如何查看覆蓋率

  • 在測試窗口下,右鍵相應的測試組 點
  • 點擊如下的"分析代碼覆蓋率"

6. 實踐中常見場景的Mock

主要

6.1 DbSet

使用EF Core過程中,如何mock DbSet是一個繞不過的坎。

方法一

參考如下鏈接的回答進行自行封裝

https://stackoverflow.com/questions/31349351/how-to-add-an-item-to-a-mock-dbset-using-moq

方法二(推薦)

使用現(xiàn)成的庫(也是基于上面的方式封裝好的)

倉庫地址:

https://github.com/romantitov/MockQueryable

使用范例

// 1. 測試時創(chuàng)建一個模擬的List<T>
var users = new List<UserEntity>()
{
 new UserEntity{LastName = "ExistLastName", DateOfBirth = DateTime.Parse("01/20/2012")},
 ...
};

// 2. 通過擴展方法轉換成DbSet<UserEntity>
var mockUsers = users.AsQueryable().BuildMock();

// 3. 賦值給給mock的DbContext中的Users屬性
var mockDbContext = new Mock<DbContext>();
mockDbContext
 .Setup(it => it.Users)
 .Return(mockUsers);

6.2 HttpClient

使用RestEase/Refit的場景

如果使用的是 RestEase 或者 Refit 等第三方庫,具體接口的定義本質上就是一個interface,所以直接使用moq進行方法mock即可。

并且建議使用這種方式。

IHttpClientFactory

如果使用的是.Net Core自帶的 IHttpClientFactory 方式來請求外部接口的話,可以參考如下的方式對 IHttpClientFactory 進行mock

https://www.thecodebuzz.com/unit-test-mock-httpclientfactory-moq-net-core/

6.3 ILogger

由于ILogger的LogError等方法都是屬于擴展方法,所以不需要特別的進行方法級別的mock。

針對平時的一些使用場景封裝了一個幫助類, 可以使用如下的幫助類進行Mock和Verify

public static class LoggerHelper
{
  public static Mock<ILogger<T>> LoggerMock<T>() where T : class
  {
    return new Mock<ILogger<T>>();
  }

  public static void VerifyLog<T>(this Mock<ILogger<T>> loggerMock, LogLevel level, string containMessage, Times times)
  {
    loggerMock.Verify(
    x => x.Log(
      level,
      It.IsAny<EventId>(),
      It.Is<It.IsAnyType>((o, t) => o.ToString().Contains(containMessage)),
      It.IsAny<Exception>(),
      (Func<It.IsAnyType, Exception, string>)It.IsAny<object>()),
    times);
  }

  public static void VerifyLog<T>(this Mock<ILogger<T>> loggerMock, LogLevel level, Times times)
  {
    loggerMock.Verify(
    x => x.Log(
      level,
      It.IsAny<EventId>(),
      It.IsAny<It.IsAnyType>(),
      It.IsAny<Exception>(),
      (Func<It.IsAnyType, Exception, string>)It.IsAny<object>()),
    times);
  }
}

使用方法

[Fact]
public void Echo_ShouldLogInformation()
{
  // arrange
  var mockLogger = LoggerHelpe.LoggerMock<UserController>();
  var controller = new UserController(mockLogger.Object);
  
  // act
  controller.Echo();
  
  // assert
  mockLogger.VerifyLog(LogLevel.Information, "hello", Times.Once());
}

7. 拓展

7.1 TDD介紹

TDD是測試驅動開發(fā)(Test-Driven Development)的英文簡稱. 一般是先提前設計好單元測試的各種場景再進行真實業(yè)務代碼的編寫,編織安全網以便將Bug扼殺在在搖籃狀態(tài)。

此種開發(fā)模式以測試先行,對開發(fā)團隊的要求較高, 落地可能會存在很多實際困難。詳細說明可以參考如下

https://www.guru99.com/test-driven-development.html

參考鏈接

https://docs.microsoft.com/en-us/dotnet/core/testing/unit-testing-best-practices

https://www.kiltandcode.com/2019/06/16/best-practices-for-writing-unit-tests-in-csharp-for-bulletproof-code/

https://github.com/AutoFixture/AutoFixture

到此這篇關于淺談.Net Core后端單元測試的實現(xiàn)的文章就介紹到這了,更多相關.Net Core 單元測試內容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!

相關文章

  • asp.net 獲取銀行貨幣匯率的代碼

    asp.net 獲取銀行貨幣匯率的代碼

    這個主要公司做外貿的每天都需要更新匯率,本來以前有一個服務可以調用,但是那個連接用不了 所以就寫了一個這樣的東西 套取網頁顯示信息
    2010-12-12
  • .net 中的 StringBuilder 和 TextWriter 區(qū)別詳解

    .net 中的 StringBuilder 和 TextWriter 區(qū)別詳解

    這篇文章主要介紹了.net 中的 StringBuilder 和 TextWriter 區(qū)別詳解,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2020-09-09
  • .net jMail郵件發(fā)送(含抄送、密送、多發(fā)、日志記錄)實例代碼

    .net jMail郵件發(fā)送(含抄送、密送、多發(fā)、日志記錄)實例代碼

    這篇文章主要介紹了.net jMail郵件發(fā)送(含抄送、密送、多發(fā)、日志記錄)實例代碼,有需要的朋友可以參考一下
    2013-11-11
  • ASP.NET中CKEditor與CKFinder的配置使用

    ASP.NET中CKEditor與CKFinder的配置使用

    這篇文章主要介紹了ASP.NET中CKEditor與CKFinder的配置使用的相關資料,需要的朋友可以參考下
    2015-06-06
  • ASP.NET小結之MVC, MVP, MVVM比較以及區(qū)別(一)

    ASP.NET小結之MVC, MVP, MVVM比較以及區(qū)別(一)

    MVC, MVP和MVVM都是用來解決界面呈現(xiàn)和邏輯代碼分離而出現(xiàn)的模式。以前只是對它們有部分的了解,沒有深入的研究過,對于一些里面的概念和區(qū)別也是一知半解?,F(xiàn)在一邊查資料,并結合自己的理解,來談一下對于這三種模式思想的理解,以及它們的區(qū)別。歡迎各位高手拍磚。
    2014-05-05
  • asp.net core 修改默認端口的幾種方法

    asp.net core 修改默認端口的幾種方法

    這篇文章主要介紹了asp.net core 修改默認端口的幾種方法,文中講解非常詳細,代碼和圖片幫助大家更好的理解,感興趣的朋友可以了解下
    2020-06-06
  • .NET 數(shù)據(jù)庫連接池

    .NET 數(shù)據(jù)庫連接池

    如果您通過使用另一個 Execute 方法(例如,ExecuteScalar、ExecuteNonQuery 和 ExecuteXMLReader)執(zhí)行查詢
    2008-12-12
  • asp.net后臺注冊js的四種方法分享

    asp.net后臺注冊js的四種方法分享

    這篇文章主要介紹了asp.net后臺注冊js的四種方法,有需要的朋友可以參考一下
    2014-01-01
  • c# indexof 用法深入理解

    c# indexof 用法深入理解

    在開發(fā)過程中經常會使用到c# indexof 查找字串中指定字符或字串首次出現(xiàn)的位置,返首索引值,本文將詳細介紹,需要的朋友可以參考下
    2012-11-11
  • 超好用輕量級MVC分頁控件JPager.Net

    超好用輕量級MVC分頁控件JPager.Net

    本文給大家分享的是一款超好用輕量級MVC分頁控件--JPager.Net,小編自己也在使用,非常的不錯,推薦給大家。
    2016-06-06

最新評論