在我们码字过程中,单元测试是必不可少的。但在从业过程中,很多开发者却对单元测试望而却步。有些时候并不是不想写,而是常常会碰到下面这些问题,让开发者放下了码字的脚步:

  1. 这个类初始数据太麻烦,你看:new MyService(new User("test",1), new MyDAO(new Connection(......)),new ToManyPropsClass(......) .....) 。我:。。。
  2. 这个代码内部逻辑都是和Cookie有关,我单元测试不好整啊,还是得启动到浏览器里一个按钮一个按钮点。
  3. 这个代码内部读了配置文件,单元测试也不能给我整个配置文件啊?
  4. 这个代码主要是验证WebAPI入口得模型绑定,必须得调用一次啊?

这些问题确实存在,但它们阻止不了我们那颗要写单元测试的心。单元测试的优点很多,你或许可以不管。但至少能让你从那些需要在浏览器里点击10多下的操作里解脱出来。本文从一个简单的逻辑测试出发,慢慢拉开测试的大幕,让你爱上测试。文章主要是传播一些单元测试的理念,其次才是介绍asp.net core中的单元测试。

本文使用的环境为asp.net core 2.1 webapi,代码可以直接下载:https://github.com/yubaolee/DotNetCoreUnitTestSamples 为了方便阅读,以一个最简单的逻辑为例:

  1. public class UserService{
  2. public bool CheckLogin(UserInfo user)
  3. {
  4. return user.Name == user.Password; //登录逻辑,为了看着舒服,少点
  5. }
  6. }
  7. public class UserInfo{
  8. public string Name { get; set; }
  9. public string Password { get; set; }
  10. }

测试的WebAPI控制器如下:

  1. public class ValuesController : ControllerBase
  2. {
  3. private UserService _service;
  4.  
  5. public ValuesController(UserService service)
  6. {
  7. _service = service;
  8. }
  9.  
  10. [HttpGet]
  11. [Route("checklogin")]
  12. public bool CheckLogin([FromQuery]UserInfo user)
  13. {
  14. return _service.CheckLogin(user);
  15. }
  16. }

都已准备完毕,那么,开始我们的表演吧:

普通业务的单元测试

  1. public class TestService
  2. {
  3. private UserService _service;
  4.  
  5. [SetUp]
  6. public void Init()
  7. {
  8. var server = new TestServer(WebHost.CreateDefaultBuilder().UseStartup<Startup>());
  9. _service = server.Host.Services.GetService<UserService>();
  10. }
  11. [Test]
  12. public void TestLogin()
  13. {
  14. bool result = _service.CheckLogin(new UserInfo { Name = "yubao", Password = "yubao" });
  15. Assert.IsTrue(result);
  16. }
  17. }

在做业务测试过程中要善于使用注入功能,而不是使用new对象的方式,比如这里的Host.Services.GetService,防止出现new MyService(new User("test",1), new MyDAO(new Connection(......)),new ToManyPropsClass(......) .....)这种尴尬。用的越多你就越能体会这种做法的好处。我在openauth.net中使用的是autofac的AutofacServiceProvider。

测试Controller

很多时候我们需要测试顶层的controller(八成是controller里混的有业务逻辑)。这时我们可以快速的写出下面的测试代码:

  1. public class TestController
  2. {
  3. private ValuesController _controller;
  4.  
  5. [SetUp]
  6. public void Init()
  7. {
  8. var server = new TestServer(WebHost.CreateDefaultBuilder().UseStartup<Startup>());
  9. _controller = server.Host.Services.GetService<ValuesController>();
  10. }
  11. [Test]
  12. public void TestLogin()
  13. {
  14. bool result = _controller.CheckLogin(new UserInfo{Name = "yubao",Password = "yubao"});
  15. Assert.IsTrue(result);
  16. }
  17. }

这段代码在JAVA spring mvc框架下是没有问题的,但在asp.net core 中,你会发现:

获取不到controller?spring mvc的理念就是万物皆服务,哪怕是一个controller也是一个普通的服务。但微软不喜欢这样,默认时它要掌控controller的生死The Subtle Perils of Controller Dependency Injection in ASP.NET Core MVC 有人在声讨微软了)。所以我们不能通过普通的ServicCollection来注入和获取它,除非你指明Controller As Service,如下:

  1. public void ConfigureServices(IServiceCollection services)
  2. {
  3. services.AddMvc().AddControllersAsServices().SetCompatibilityVersion(CompatibilityVersion.Version_2_1);
  4. }

这时即可顺利测试通过。

测试含有HTTP上下文的业务逻辑,比如Cookie、URL中的QueryString

在平时的代码过程中,常常会和HTTP上下文HttpContext打交道,最常见的如request、response、cookie、querystring等,比如我们新的逻辑:

  1. public class UserService
  2. {
  3. private IHttpContextAccessor _httpContextAccessor;
  4.  
  5. public UserService(IHttpContextAccessor httpContextAccessor)
  6. {
  7. _httpContextAccessor = httpContextAccessor;
  8. }
  9.  
  10. public bool IsLogin()
  11. {
  12. return _httpContextAccessor.HttpContext.Request.Cookies["username"] != null;
  13. }
  14. }

这时如何测试呢?马丁福勒在他的大作《企业应用架构模式》中明确指出“测试桩”的概念,来应对这种情况。各种Mock框架应运而生。比如我最喜欢的Moq:

  1. public class TestCookie
  2. {
  3. private UserService _service;
  4.  
  5. [SetUp]
  6. public void Init()
  7. {
  8. var httpContextAccessorMock = new Mock<IHttpContextAccessor>();
  9. httpContextAccessorMock.Setup(x => x.HttpContext.Request.Cookies["username"]).Returns("yubaolee");
  10.  
  11. var server = new TestServer(WebHost.CreateDefaultBuilder()
  12. .ConfigureServices(u =>u.AddScoped(x =>httpContextAccessorMock.Object))
  13. .UseStartup<Startup>());
  14. _service = server.Host.Services.GetService<UserService>();
  15. }
  16. [Test]
  17. public void TestLogin()
  18. {
  19. bool result = _service.IsLogin();
  20. Assert.IsTrue(result);
  21. }
  22. }

测试一次HTTP请求

有时我们需要测试Mvc框架的模型绑定,看看一次客户端的请求是否能被正确解析,亦或者测试WebAPI入口的一些Filter AOP等是否被正确触发,这时就需要测试一次HTTP请求。从严格意义上来讲这种测试已经脱离的单元测试的范畴,属于集成测试。但这种测试代码可以节省我们大量的重复劳动。asp.net core中可以通过TestServer快速实现这种模拟:

  1. public class TestHttpRequest
  2. {
  3. private TestServer _testServer;
  4.  
  5. [SetUp]
  6. public void Init()
  7. {
  8. _testServer = new TestServer(WebHost.CreateDefaultBuilder().UseStartup<Startup>());
  9. }
  10. [Test]
  11. public void TestLogin()
  12. {
  13. var client = _testServer.CreateClient();
  14. var result = client.GetStringAsync("/api/values/checklogin?name=yubao&password=yubao");
  15. Console.WriteLine(result.Result);
  16. }
  17. }

在进行单元测试的过程中,测试的理念(或者TDD的思维?)异常重要,它能帮助你构建和谐优美的代码。

G

关于单元测试的思考--Asp.Net Core单元测试最佳实践的更多相关文章

  1. 【转】.NET(C#):浅谈程序集清单资源和RESX资源 关于单元测试的思考--Asp.Net Core单元测试最佳实践 封装自己的dapper lambda扩展-设计篇 编写自己的dapper lambda扩展-使用篇 正确理解CAP定理 Quartz.NET的使用(附源码) 整理自己的.net工具库 GC的前世与今生 Visual Studio Package 插件开发之自动生

    [转].NET(C#):浅谈程序集清单资源和RESX资源   目录 程序集清单资源 RESX资源文件 使用ResourceReader和ResourceSet解析二进制资源文件 使用ResourceM ...

  2. Asp.Net Core 单元测试正确姿势

    背景 ASP.NET Core 支持依赖关系注入 (DI) 软件设计模式,并且默认注入了很多服务,具体可以参考 官方文档, 相信只要使用过依赖注入框架的同学,都会对此有不同深入的理解,在此无需赘言. ...

  3. 使用 xUnit 编写 ASP.NET Core 单元测试

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

  4. 基于Jenkins Pipeline的ASP.NET Core持续集成实践

    最近在公司实践持续集成,使用到了Jenkins的Pipeline来提高团队基于ASP.NET Core API服务的集成与部署效率,因此这里总结一下. 一.关于持续集成与Jenkins Pipelin ...

  5. ASP.NET Core MVC TagHelper实践HighchartsNET快速图表控件-开源

    ASP.NET Core MVC TagHelper最佳实践HighchartsNET快速图表控件支持ASP.NET Core. 曾经在WebForms上写过 HighchartsNET快速图表控件- ...

  6. asp.net core的docker实践

    如果centos中没有安装和docker和.net core镜像,先安装docker和asp.net core 镜像 安装dockeryum -y install docker-io 启动 Docke ...

  7. 使用Docker部署ASP.NET Core应用程序实践

    前言 最近把很火的Docker给看了,于是就磨拳擦掌要去实践一下.于是就拿之前一个aps.net core的项目(已被停止)去练手.该项目之前在ubuntu14.04上确保可以正常运行,所以docke ...

  8. Dependency injection in .NET Core的最佳实践

    我们知道依赖注入(DI)是一种实现对象及其协作者或依赖关系之间松散耦合的技术. ASP.NET Core包含一个简单的内建容器来支持构造器注入. 我们试图将DI的最佳实践带到.NET Core应用程序 ...

  9. [转]ASP.NET MVC 4 最佳实践宝典

    原文:http://www.cnblogs.com/sonykings/archive/2013/05/30/3107531.html ASP.NET MVC最佳实践 本文档提供了一套旨在帮助创建最佳 ...

随机推荐

  1. Scrapy 1.4 文档 01 初窥 Scrapy

    初窥 Scrapy Scrapy 是用于抓取网站并提取结构化数据的应用程序框架,其应用非常广泛,如数据挖掘,信息处理或历史存档. 尽管 Scrapy 最初设计用于网络数据采集(web scraping ...

  2. react,react native,webpack,ES6,node.js----------今天上午学了一下node.js

    http://www.yiibai.com/nodejs/node_install.html---node.js具体入门资料在此 Node JS事件循环 Node JS是单线程应用程序,但它通过事件和 ...

  3. Docker 新手入门

    简介 如果您是 Docker 新手请您花大约三十分钟的时间来了解 Docker 相关的知识和内容. Docker 与 Linux 息息相关,因此在阅读本文档之前请您确保以下条件: 对 Linux 的命 ...

  4. index_levedb.go

    )     binary.BigEndian.PutUint64(key, fid)     return l.db.Delete(key, nil) } //关闭资源 func (l *LevelD ...

  5. upload.go

    package api import (     "os"     "bytes"     "mime/multipart"     &qu ...

  6. Dubbo中暴露服务的过程解析

    dubbo暴露服务有两种情况,一种是设置了延迟暴露(比如delay="5000"),另外一种是没有设置延迟暴露或者延迟设置为-1(delay="-1"): 设置 ...

  7. vue2.0 axios封装、vuex介绍

    一.前言 博主也是vue道路上的行者,道行不深,希望自己的东西能对大家有所帮助.这篇博客针对 了解过vue基础,但是没有做过vue项目的童鞋.如果想看基础指令,可以看我之前的一篇博客,请点击  跳转, ...

  8. Java开发者必备的10大学习网站,送给入门学习java的你,请收下!

    作为开发者来说,必备的除了对编码的热情还要有自己的一套技巧,另外不可缺少的就是平时学习的网站.以下本人收集的 Java 开发者必备的网站,这些网站可以提供信息.以及一些很棒的讲座 , 还能解答一般问题 ...

  9. 大白话5分钟带你走进人工智能-第十四节过拟合解决手段L1和L2正则

                                                                               第十四节过拟合解决手段L1和L2正则 第十三节中, ...

  10. Javapoet源码解析

    Javapoet:是生成.java源文件的开源API  github:https://github.com/square/javapoet   以生成一个HelloWrold.java文件为例: pa ...