好代码是管出来的——.Net Core中的单元测试与代码覆盖率

 测试对于软件来说,是保证其质量的一个重要过程,而测试又分为很多种,单元测试、集成测试、系统测试、压力测试等等,不同的测试的测试粒度和测试目标也不同,如单元测试关注每一行代码,集成测试关注的是多个模块是否能正常的协同工作。
  当我们在衡量代码好坏时,其中一点就是这些代码是否进行了单元测试,测试的质量、代码覆盖率怎么样?本文将从以下几个方面介绍.Net Core中的单元测试:

  • 单元测试简介

  • .Net Core中的单元测试框架

  • 使用xUnit.Net对.Net Core应用进行单元测试

    • 创建xUnit.Net测试项目

    • 编写测试方法

    • 断言

    • 运行单元测试

  • Mock

  • 单元测试代码覆盖率

  • 小结

单元测试简介

  单元测试是指对软件中的最小可测试单元进行检查验证,而.Net中最小可测试单元就是类和方法,单元测试是白盒测试关注于代码执行逻辑,所以单元测试代码一般也由开发人员编写。
  单元测试关注的两个重点就是最小可测试单元和代码逻辑,但是很多情况下,一个类或者是方法它会依赖一些外部组件,如其他开发人员写的代码、第三方类库、数据库、网络等,当被测试的代码与这些组件紧耦合时,那么这段代码将可能是不可测的,如一个方法中依赖一个数据库组件去访问数据库,那么在执行这个方法时,必然要与数据库交互,如果没有数据库,那么该方法就无法运行。
  所以单元测试不仅是对代码逻辑进行检查,同时还对整个代码结构有所限制,面向对象编程时应当遵循“依赖倒置”原则,模块应该依赖抽象,抽象不应该依赖实现。并且所依赖的抽象,应该显示的通过构造或者方法参数进行暴露,让组件的使用者对组件的依赖一目了然。
  而在单元测试时为了屏蔽这些抽象依赖,不同测试框架中提供了stub、mock、fake等方式对抽象进行模拟,以便于代码能够正常执行。

.Net Core中的单元测试框架

  .Net Core中常用的单元测试的框架有MSTest、NUnit和xUnit.net,它们的使用方法都非常相似,都是通过特性标记的方式声明测试方法,然后在方法中使用断言(Assertions)来判别方法执行结果是否达到预期。
  这三个框架中MSTest是与VS集成的,而NUnit和xUnit.net都加入了.Net基金会,下面两个图分别是三个框架特性和断言的比较(内容来自:https://xunit.github.io/docs/comparisons):
  特性:

  640?wx_fmt=png

  断言(部分):

  640?wx_fmt=png 

  三个框架自有优点,但xUnit.net使用更广泛一些(许多开源项目都使用xUnit.net,包括ASP.NET Core MVC、EF Core等项目),支持.Net下的大部分平台(.Net Fx、.Net Core、UWP、Xamarin),并且具有非常好的可拓展性。

使用xUnit.Net对.Net Core应用进行单元测试

  本文使用xUnit.Net框架来对.Net Core程序进行单元测。

创建xUnit.Net测试项目

  在解决方案中添加.Net Core的被测试项目以及xUnit测试项目:

  640?wx_fmt=png

  目录结构:

  640?wx_fmt=png

  xUnit测试项目还提供了相应的代码分析器来帮助编写测试代码:

  640?wx_fmt=png

编写测试方法

  在被测试的项目中添加一个计算器类型,并添加加法运算的方法:

   640?wx_fmt=png

  在测试项目中添加Calulator方法的测试代码:

  640?wx_fmt=png

断言

  在程序中,断言指的是一个表达式语句执行时总是为真(True),它有助于代码阅读、调试、编译和缺陷检测,当断言内表达式执行为True时,不会执行任何操作,当结果为false时将会输出一些异常信息,下图是.Net中System.Diagnositics命名空间下提供的代码调试使用的断言用法(在调试程序时,当参数x < y就会中断并抛出异常信息):

  640?wx_fmt=png

  更多参考:https://docs.microsoft.com/en-us/visualstudio/debugger/assertions-in-managed-code
  在单元测试中,测试方法也是使用断言的方式来判别程序执行结果与预期结果是否相符:

  640?wx_fmt=png

  xUnit.net中的断言参考:https://github.com/xunit/assert.xunit

运行单元测试

  在VS中可以使用VS的测试窗口运行测试方法:

  640?wx_fmt=jpeg

  运行结果(测试通过):

  640?wx_fmt=png

  运行结果(测试未通过):

  640?wx_fmt=png

Mock

  在文章前面提到过,面向对象编程应该显示的依赖抽象,单元测试时应该将屏蔽依赖的影响(无论是依赖还未实现,或者实现的依赖会阻碍代码执行),为了满足这一需求出现了Mock、Fake等方式,其原理就是创建一个"假"的"空"的依赖,并用其替代真实依赖,以确保代码能够运行。
  .Net中一个常用的Mock框架是Moq,本文将使用Moq来介绍如何对依赖进行模拟:
  1. 编写需要依赖的代码:

  640?wx_fmt=png

  640?wx_fmt=png

  上面代码中UserManager依赖一个用户的仓储类型,该仓储将会与数据库交互。
  2. 为测试项目安装Moq组件:

  640?wx_fmt=jpeg

  3. 编写测试代码:

  640?wx_fmt=png

  上面代码通过Moq组件Mock了一个IUserRepository的类型,并将其Add方法设置并返回true(注:设置方法时参数的数据要与调用时使用的一致),最后通过Mock的对象实例Object来创建UserManager实例。
  最后断言当创建用户时,年龄为负数则抛出FormatException。
  4. 运行测试:

  640?wx_fmt=png

  测试成功。

单元测试代码覆盖率

  测试代码覆盖率是对单元测试的一种度量,可以用来衡量单元测试是否达标,一般将代码测试目标定到80%-90%之间,为了保证代码覆盖率,在写测试用例时就要从语句覆盖、条件覆盖、路径覆盖等方面进行充分考虑。
  而.Net Core中如何在测试时计算代码覆盖率呢?如果使用VS的企业版,那么VS自带了代码覆盖率分析工具:

  640?wx_fmt=png 

  详情参考:https://docs.microsoft.com/en-us/visualstudio/test/using-code-coverage-to-determine-how-much-code-is-being-tested
       https://github.com/Microsoft/vstest-docs/blob/master/docs/analyze.md#coverage
  注:VS集成了MSTest,所以代码覆盖分析工具对MSTest支持非常好,但对xUnit.Net的支持如何笔者未进行测试。

  对于xUnit.net来说,要分析测试代码覆盖率还可以通过“OpenCover”和“ReportGenerator”工具完成,下面就介绍如何通过这两个工具完成代码覆盖率的分析:
  1. 下载并安装OpenCover,在OpenCover的GitHub上下载最新release的zip包,并解压缩到指定目录下,并将OpenCover目录添加到环境变量中:

  640?wx_fmt=png

  地址:https://github.com/OpenCover/opencover/releases

  2. 通过命令行使用OpenCover来完成覆盖率分析:
  OpenCover有许多参数,具体参考:https://github.com/OpenCover/opencover/wiki/Usage
  在本例中,仅需要指定目标程序是dotnet.exe,目标程序参数是test(注:.Net Core的测试功能实际上是用.Net Core的CLI命令 dotnet test完成的),另外指定输出文件名,register参数用于注册代码分析器默认使用user即可,-filter参数用于过滤不需要分析覆盖率的程序集和类型,-oldstyle是为了支持.Net Core程序添加的参数(详见:https://github.com/OpenCover/opencover/issues/595)
  另外为了能够满足测试需要在相关项目文件中添加以下节点(详见:https://github.com/Microsoft/vstest/issues/800):

<PropertyGroup><DebugType>full</DebugType></PropertyGroup>

  最后在项目目录下执行以下命令:
  OpenCover.Console.exe -target:"dotnet.exe" -targetargs:"test" -output:coverage.xml -register:user -filter:"+[*]* -[*Moq]* -[xunit*]*" -oldstyle

  640?wx_fmt=png

  生成的结果: 

  4. 通过ReportGenerator生成可读报表:
  下载地址:https://github.com/danielpalme/ReportGenerator/releases
  注:下载解压后将ReprotGenerator的目录添加到环境变量,以便使用。
  执行以下命令:
  ReportGenerator.exe "-reports:coverage.xml" "-targetdir:report"

  640?wx_fmt=png

  生成内容:

  640?wx_fmt=png

  打开index.htm文件:

  640?wx_fmt=png

  5. 在项目中创建一个bat文件,用于保存代码覆盖率检测和报表生成命令,便于使用:

  640?wx_fmt=png

小结

  本文主要介绍了如何使用xUnit.net测试框架完成.Net Core程序的单元测试,以及通过Moq框架来模拟测试目标的相关依赖,避免了其它组件对测试代码的影响。
  文章的最后介绍了如何使用开源工具OpenCover和ReportGenerator工具来实现.Net Core单元测试代码覆盖率分析,这种方案相对VS企业版自带的工具使用上要麻烦一些,但好在工具都是开源的,对持续集成也有比较好的支持,所以不失为一种好的解决方案。
  单元测试仅能保证软件的最小可执行单元是正确的,真正的软件是由这些最小可执行单元组成的一个整体,单元的正确性无法保证整体的正确性,下篇文章将对.Net Core的集成测试进行介绍。

本文链接:https://www.cnblogs.com/selimsong/p/9263957.html 

测试代码:https://github.com/yqszt/xUnitTestDemo

参考:
  https://en.wikipedia.org/wiki/Unit_testing
  https://docs.microsoft.com/en-us/dotnet/core/testing/
  https://github.com/aspnet/Home/wiki/Engineering-guidelines#unit-tests-and-functional-tests
  http://asp.net-hacker.rocks/2017/03/31/unit-testing-with-dotnetcore.html
  https://stackoverflow.com/questions/261139/nunit-vs-mbunit-vs-mstest-vs-xunit-net
  http://blog.ploeh.dk/2010/04/26/WhyImmigratingfromMSTesttoxUnit.net/
  https://docs.microsoft.com/en-us/dotnet/core/tools/dotnet-test?tabs=netcore21
  https://stackoverflow.com/questions/38425936/how-to-measure-code-coverage-in-asp-net-core-projects-in-visual-studio
  http://dotnetliberty.com/index.php/2016/02/22/moq-on-net-core/
  https://github.com/moq/moq4
  https://stackoverflow.com/questions/41384459/opencover-reports-missing-pdbs-when-pdbs-are-present-xunit-net-core
  https://github.com/OpenCover/opencover
  https://github.com/danielpalme/ReportGenerator

相关文章:

原文地址https://www.cnblogs.com/selimsong/p/9263957.html

.NET社区新闻,深度好文,欢迎访问公众号文章汇总 http://www.csharpkit.com

640?wx_fmt=jpeg

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/320916.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

数字图像处理作业

图像处理作业 1 取sT(r)11(mr)EsT(r)\frac{1}{1(\frac{m}{r})^E}sT(r)1(rm​)E1​ 其中rrr为原始亮度&#xff0c;mmm为输入区间的中点&#xff0c;EEE描述曲线的陡峭程度 2 一幅8灰度级图像具有如下所示的直方图&#xff0c;求直方图均衡后的灰度级和对应概率&#xff0c;…

深港澳大湾区(深圳).NET技术交流会圆满成功

2018年7月7日一场以.NET Core微服务和机器学习为主题的交流会成功在深圳职业技术学院落下帷幕。这次活动在短短的一周时间内&#xff0c;报名人数超过了170人&#xff0c;除了一些同学临时有事&#xff0c;基本都到现场了&#xff0c;特别感谢深职院的软创工作室对这次活动的支…

.Net Core开发日志——Peachpie

.Net Core的生态圈随着开源社区的力量不断注入至其中&#xff0c;正在变得越来越强盛&#xff0c;并且不时得就出现些有意思的项目&#xff0c;比如Peachpie&#xff0c;它使得PHP的代码迁移到.Net Core项目变得可能。从创建简单的入门程序开始可以更容易地体会其特性。首先安装…

.NET Core开发日志——Middleware

熟悉ASP.NET架构的开发者一定对于HTTP Modules与HTTP Handlers不陌生。两者的作用主要是对网络请求执行特定的处理工作。而在.NET Core中&#xff0c;它们都被Middleware(中件间)取代了。之前的Http Modules和HTTP Handlers是如下图般处理请求的&#xff1a;现在变成了这样&…

Identity Server 4 - Hybrid Flow - Claims

前一篇 Identity Server 4 - Hybrid Flow - MVC客户端身份验证: https://www.cnblogs.com/cgzl/p/9253667.htmlClaims我不知道怎么样翻译这个词比较好, 所以我一般就不翻译了.在前一篇文章里, MVC客户端配置身份认证的时候有这么一句话(Startup的ConfigureServices):JwtSecurit…

图像处理作业第7次

图像处理作业第7次 1.请根据课本中Z变换的定义&#xff0c;证明如下结论。 (1)若x(n)x(n)x(n)的ZZZ变换为X(z)X(z)X(z)&#xff0c;则(−1)nx(n)(-1)^nx(n)(−1)nx(n)的ZZZ变换为X(−z)X(-z)X(−z) 根据ZZZ变换的定义 X(z)∑x(n)z−n,∑(−1)nx(n)z−n∑x(n)(−z)−nX(−z)X(z…

微软宣布ASP.NET Core 2.0正式支持OData标准

近日&#xff0c;OData 团队在微软开发者博客上宣布&#xff0c;ASP.NET Core 2.0 已正式支持 OData 标准&#xff0c;开发者现在可通过包管理器 NuGet 来获取 Microsoft.AspNetCore.OData 包。此软件包包含在使用 ASP.NET Core MVC 时创建 OData v4.0 端点以及支持 Web API 的…

图像处理作业4

图像处理作业4 1. 第二版课本习题4.21 本质没有区别&#xff0c;只将图片放置在中心&#xff0c;而周围填充0的个数不变时&#xff0c;不会影响结果。因为本质都是进行了周期延拓&#xff0c;使得尾部的信息不会被丢弃掉。相当于滤波前将图像进行了平移。需要注意的是&#x…

[译]ASP.NET Core Web API 中使用Oracle数据库和Dapper看这篇就够了

园子里关于ASP.NET Core Web API的教程很多&#xff0c;但大多都是使用EFMysql或者EFMSSQL的文章。甚至关于ASP.NET Core Web API中使用DapperMysql组合的文章都很少&#xff0c;更别提OracelDapper组合的文章了&#xff0c;那么今天就带着大家一起翻译一篇国外大牛写的关于ASP…

一个c#开发的web绘流系统

本项目为Web版流程图绘图&#xff0c;兼容IE、FF、Chrome等各主流浏览器&#xff0c;提供了各类基础图形&#xff0c;至于如何应用及用来做什么就随各位了&#xff0c;常规的流程图及逻辑辅助、工作流等皆可胜任。本系统虽为开源项目&#xff0c;然精心调配开发测试&#xff0c…

将系统分解为微服务的策略

几年前&#xff0c;Vladik Khononov和他的团队决定开始使用微服务&#xff0c;但是几个月后他们发现自己陷入了巨大的混乱之中。他在最近于伦敦Skills Matter举行的DDD eXchange 2018会议上指出&#xff0c;造成这一现象的原因在于&#xff0c;他们只专注于采用酷炫的新技术&am…

.NET Core微服务之基于Ocelot+IdentityServer实现统一验证与授权

一、案例结构总览这里&#xff0c;假设我们有两个客户端&#xff08;一个Web网站&#xff0c;一个移动App&#xff09;&#xff0c;他们要使用系统&#xff0c;需要先向IdentityService进行Login以进行验证并获取Token&#xff0c;在IdentityService的验证过程中会访问数据库以…

CentOS安装使用.netcore极简教程(免费提供学习服务器)

本文目标是指引从未使用过Linux的.Neter&#xff0c;如何在CentOS7上安装.Net Core环境&#xff0c;以及部署.Net Core应用。 仅针对CentOS&#xff0c;其它Linux系统类似&#xff0c;命令环节稍加调整&#xff1b;需要提前准备好服务器地址、用户名、密码&#xff1b;如果手上…

ASP.NET Core MVC+EF Core从开发到部署

笔记本电脑装了双系统&#xff08;Windows 10和Ubuntu16.04&#xff09;快半年了&#xff0c;平时有时间就喜欢切换到Ubuntu系统下耍耍Linux&#xff0c;熟悉熟悉Linux命令、Shell脚本以及Linux下的各种应用的安装、配置、运行。使用下来的感受是Linux确实相当好使&#xff0c;…

网络流及建模专题(上)

前言 不断更新中…… 这几天新坑填不下去了&#xff0c;回来回顾一些经典的模型套路&#xff0c;先拿网络流开刀&#xff0c;窃以为洛谷这几道网络流的题目还是非常具有代表性的&#xff0c;涵盖了网络流调整、多解计数、最小割、最大权闭合子图问题。 还涵盖了图论&#xff0…

基于 websocket 实现的 im 实时通讯案例

分享利用 redis 订阅与发布特性&#xff0c;巧妙的现实高性能im系统。为表诚意&#xff0c;先贴源码地址&#xff1a;https://github.com/2881099/im下载源码后的运行方法&#xff1a;运行环境&#xff1a;.NETCore 2.1 redis-server 2.8下载Redis-x64-2.8.2402.zip&#xff0…

ACM/ICPC 比赛生涯总结+经验分享

ACM/ICPC 比赛生涯总结经验分享 1.获奖经历 时间比赛奖励大一下ACM陕西省赛打铁大一下CCCC团队二等奖大二下ACM/ICPC全国邀请赛银奖大二下CCCC团队特等奖大三上ACM/ICPC区域赛沈阳站铜奖大三上ACM/ICPC区域赛南宁站银奖大三上ACM/ICPC EC-Final上海铜奖大三下CCCC团队特等奖大…

NCC Meetup 2018 Shanghai 活动小结

NCC Meetup 2018 上海的活动于2018年6月30日在微软上海港汇办公室进行。原本计划30人规模的小型活动&#xff0c;结果收到了逾60人的报名&#xff0c;其中大部均来到现场参加了活动。本次活动得到了微软公司的场地支持&#xff0c;同时非常感谢 范亮先生、 刘浩杨先生和 邹嵩…

Asp.Net Core 使用Quartz基于界面画接口管理做定时任务

今天抽出一点点时间来造一个小轮子&#xff0c;是关于定时任务这块的。这篇文章主要从一下几点介绍&#xff1a;创建数据库管理表创建web项目引入quarzt nuget 包写具体配置操作&#xff0c;实现定时任务处理第一步&#xff1a;创建一个空web项目&#xff0c;引入quarzt nuget …

.NET Core微服务之服务间的调用方式(REST and RPC)

一、REST or RPC ?1.1 REST & RPC微服务之间的接口调用通常包含两个部分&#xff0c;序列化和通信协议。常见的序列化协议包括json、xml、hession、protobuf、thrift、text、bytes等&#xff1b;通信比较流行的是http、soap、websockect&#xff0c;RPC通常基于TCP实现&am…