我的编程空间,编程开发者的网络收藏夹
学习永远不晚

Net Core后端单元测试的实现方法

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

Net Core后端单元测试的实现方法

这篇文章主要介绍了Net Core后端单元测试的实现方法,具有一定借鉴价值,感兴趣的朋友可以参考下,希望大家阅读完这篇文章之后大有收获,下面让小编带着大家一起了解一下。

1. 前言

单元测试一直都是"好处大家都知道很多,但是因为种种原因没有实施起来"的一个老大难问题。具体是否应该落地单元测试,以及落地的程度, 每个项目都有自己的情况。

本篇为个人认为"如何更好地写单元测试", 即更加 偏向实践向 中夹杂一些理论的分享。

下列示例的单元测试框架为 xUnit , Mock库为 Moq

2. 为什么需要单元测试

优点有很多, 这里提两点我个人认为的很明显的好处

2.1 防止回归

通常在进行新功能/模块的开发或者是重构的时候,测试会进行回归测试原有的已存在的功能,以验证以前实现的功能是否仍能按预期运行。

使用单元测试,可在每次生成后,甚至在更改一行代码后重新运行整套测试, 从而可以很大程度减少回归缺陷。

2.2 减少代码耦合

当代码紧密耦合或者一个方法过长的时候,编写单元测试会变得很困难。当不去做单元测试的时候,可能代码的耦合不会给人感觉那么明显。为代码编写测试会自然地解耦代码,变相提高代码质量和可维护性。

3. 基本原则和规范

 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 尽量避免直接测试私有方法

尽管私有方法可以通过反射进行直接测试,但是在大多数情况下,不需要直接测试私有的private方法, 而是通过测试公共public方法来验证私有的private方法。

可以这样认为:private方法永远不会孤立存在。更应该关心的是调用private方法的public方法的最终结果。

3.3 重构原则

如果一个类/方法,有很多的外部依赖,造成单元测试的编写困难。那么应该考虑当前的设计和依赖项是否合理。是否有部分可以存在解耦的可能性。选择性重构原有的方法,而不是硬着头皮写下去.

4 避免多个断言

如果一个测试方法存在多个断言,可能会出现某一个或几个断言失败导致整个方法失败。这样不能从根本上知道是了解测试失败的原因。

所以一般有两种解决方案

  • 拆分成多个测试方法

  • 使用参数化测试, 如下示例

[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 文件和方法命名规范 文件名规范

一般有两种。比如针对 UserController 下方法的单元测试应该统一放在 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)使用的单元测试框架

三大测试框架发展至今已是大差不差, 很多时候选择只是靠个人的喜好。

个人偏好 xUnit 简洁的断言

// xUnitAssert.True()Assert.Equal()// MsTestAssert.IsTrue()Assert.AreEqual()

客观地功能性地分析三大框架地差异可以参考如下

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

4.2 Moq

官方仓库

https://github.com/moq/moq4

Moq是一个非常流行的模拟库, 只要有一个接口它就可以动态生成一个对象, 底层使用的是Castle的动态代理功能.

基本用法

在实际使用中可能会有如下场景

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: 当入参为233时返回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方法一次,且入参为233  mockUserService.Verify(it => it.GetUser(233), Times.AtMostOnce());}

4.3 AutoFixture

官方仓库

https://github.com/AutoFixture/AutoFixture

AutoFixture是一个假数据填充库,旨在最小化3A中的 arrange 阶段,使开发人员更容易创建包含测试数据的对象,从而可以更专注与测试用例的设计本身。

基本用法

直接使用如下的方式创建强类型的假数据

[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版本,社区版不带这个功能。

如何查看覆盖率

  • 在测试窗口下,右键相应的测试组 点

  • 点击如下的"分析代码覆盖率"

Net Core后端单元测试的实现方法

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

Net Core后端单元测试的实现方法

方法二(推荐)

使用现成的库(也是基于上面的方式封装好的)

仓库地址:

https://github.com/romantitov/MockQueryable

使用范例

// 1. 测试时创建一个模拟的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是测试驱动开发(Test-Driven Development)的英文简称. 一般是先提前设计好单元测试的各种场景再进行真实业务代码的编写,编织安全网以便将Bug扼杀在在摇篮状态。

Net Core后端单元测试的实现方法

此种开发模式以测试先行,对开发团队的要求较高, 落地可能会存在很多实际困难。详细说明可以参考如下

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

感谢你能够认真阅读完这篇文章,希望小编分享的“Net Core后端单元测试的实现方法”这篇文章对大家有帮助,同时也希望大家多多支持编程网,关注编程网行业资讯频道,更多相关知识等着你来学习!

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

Net Core后端单元测试的实现方法

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

Net Core后端单元测试的实现方法

这篇文章主要介绍了Net Core后端单元测试的实现方法,具有一定借鉴价值,感兴趣的朋友可以参考下,希望大家阅读完这篇文章之后大有收获,下面让小编带着大家一起了解一下。1. 前言单元测试一直都是"好处大家都知道很多,但是因为种种原因没有实施
2023-06-08

.NET Core单元测试的方法有哪些

这篇文章主要介绍“.NET Core单元测试的方法有哪些”的相关知识,小编通过实际案例向大家展示操作过程,操作方法简单快捷,实用性强,希望这篇“.NET Core单元测试的方法有哪些”文章能帮助大家解决问题。应用程序测试的类型很多,包括集成
2023-06-30

xUnit 编写 ASP.NET Core 单元测试的方法

还记得 .NET Framework 的 ASP.NET WebForm 吗?那个年代如果要在 Web 层做单元测试简直就是灾难啊。.NET Core 吸取教训,在设计上考虑到了可测试性,就连 ASP.NET Core 这种 Web 或 A
2022-06-07

SpringBoot+TestNG单元测试的实现方法

这篇文章主要讲解了“SpringBoot+TestNG单元测试的实现方法”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“SpringBoot+TestNG单元测试的实现方法”吧!目录背景接口测
2023-06-20

Android 单元测试的主要检测方法

Android 单元测试 可以自检代码的准确性,避免重大错误流到后续工序中,也是编程的良好习惯。Andorid 单元测试主要检测方法是否正确,主要步骤:1. 在AndroidManifest.xml中加入以下代码:
2022-06-06

怎么实现Python的add5()单元测试

本篇内容介绍了“怎么实现Python的add5()单元测试”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!TestAdd5类由unittest
2023-06-17

GO中的单元测试怎么实现

这篇“GO中的单元测试怎么实现”文章的知识点大部分人都不太理解,所以小编给大家总结了以下内容,内容详细,步骤清晰,具有一定的借鉴价值,希望大家阅读完这篇文章能有所收获,下面我们一起来看看这篇“GO中的单元测试怎么实现”文章吧。
2023-07-04

基于Android系统的单元测试方法

Android单元测试如果想在android里面做单元测试,有两条基本的路子可行。第一, 是java程序员为熟悉和常用的JUnit, 但是由于目前android sdk (version 1.1)中只是提供了stubbed methods/
2022-06-06

C++ 函数单元测试的替代方法?

除 google test 外,c++++ 单元测试还有其他现代且灵活的方法,包括:catch2:现代、轻量级的框架,易于使用和配置。doctest:无头文件依赖关系,直接包含即可使用。boost.test:功能丰富,提供异常测试和 moc
C++ 函数单元测试的替代方法?
2024-04-24

Golang端口复用测试的实现方法

小编给大家分享一下Golang端口复用测试的实现方法,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下面让我们一起去了解一下吧!先给出结论:同一个进程,使用一个端口,然后连接关闭,大约需要30
2023-06-14

Golang实现单元测试中的逻辑层

前面我们完成了最麻烦的数据层的单元测试,今天我们来看看单元测试中最容易做的一层,数据逻辑层,也就是我们通常说的service或者biz等
2023-03-10

Golang实现单元测试中的接口层

接口层主要负责的就是请求的处理,最常见的就是 HTTP 请求的处理。这篇文章主要为大家介绍了Golang如何实现单元测试中的接口层,需要的可以参考一下
2023-03-11

Go 中异步函数的单元测试方法

在 go 中,异步函数可以通过并发测试进行单元测试,以模拟并发执行并测试异步函数的行为。步骤如下:创建一个超时上下文。创建一个通道来接收结果。调用异步函数并将结果写入通道。从通道中读取结果并检查预期值。使用 select 语句处理超时或接收
Go 中异步函数的单元测试方法
2024-05-01

Go 函数单元测试的自动化方法

在 go 中,可以通过以下方法自动化单元测试:使用 go test 命令,提供灵活的测试运行和管理选项。集成第三方测试框架,获得额外的功能和灵活性。利用持续集成工具,在每次代码更改时自动运行测试。Go 函数单元测试的自动化方法在 Go 中
Go 函数单元测试的自动化方法
2024-05-03

编程热搜

  • Python 学习之路 - Python
    一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
    Python 学习之路 - Python
  • chatgpt的中文全称是什么
    chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
    chatgpt的中文全称是什么
  • C/C++中extern函数使用详解
  • C/C++可变参数的使用
    可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
    C/C++可变参数的使用
  • css样式文件该放在哪里
  • php中数组下标必须是连续的吗
  • Python 3 教程
    Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
    Python 3 教程
  • Python pip包管理
    一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
    Python pip包管理
  • ubuntu如何重新编译内核
  • 改善Java代码之慎用java动态编译

目录