XUnit 依赖注入

XUnit 依赖注入

Intro

现在的开发中越来越看重依赖注入的思想,微软的 Asp.Net Core 框架更是天然集成了依赖注入,那么在单元测试中如何使用依赖注入呢?

本文主要介绍如何通过 XUnit 来实现依赖注入, XUnit 主要借助 SharedContext 来共享一部分资源包括这些资源的创建以及释放。

Scoped

针对 Scoped 的对象可以借助 XUnit 中的 IClassFixture 来实现

  1. 定义自己的 Fixture,需要初始化的资源在构造方法里初始化,如果需要在测试结束的时候释放资源需要实现 IDisposable 接口

  2. 需要依赖注入的测试类实现接口 IClassFixture<Fixture>

  3. 在构造方法中注入实现的 Fixture 对象,并在构造方法中使用 Fixture 对象中暴露的公共成员

Singleton

针对 Singleton 的对象可以借助 XUnit 中的 ICollectionFixture 来实现

  1. 定义自己的 Fixture,需要初始化的资源在构造方法里初始化,如果需要在测试结束的时候释放资源需要实现 IDisposable 接口

  2. 创建 CollectionDefinition,实现接口 ICollectionFixture<Fixture>,并添加一个 [CollectionDefinition("CollectionName")] Attribute, CollectionName 需要在整个测试中唯一,不能出现重复的 CollectionName

  3. 在需要注入的测试类中添加 [Collection("CollectionName")] Attribute,然后在构造方法中注入对应的 Fixture

Tips

  • 如果有多个类需要依赖注入,可以通过一个基类来做,这样就只需要一个基类上添加 [Collection("CollectionName")] Attribute,其他类只需要集成这个基类就可以了

Samples

Scoped Sample

这里直接以 XUnit 的示例为例:

public class DatabaseFixture : IDisposable	
{	public DatabaseFixture()	{	Db = new SqlConnection("MyConnectionString");	// ... initialize data in the test database ...	}	public void Dispose()	{	// ... clean up test data from the database ...	}	public SqlConnection Db { get; private set; }	
}	
public class MyDatabaseTests : IClassFixture<DatabaseFixture>	
{	DatabaseFixture fixture;	public MyDatabaseTests(DatabaseFixture fixture)	{	this.fixture = fixture;	}	[Fact]	public async Task GetTest()	{	// ... write tests, using fixture.Db to get access to the SQL Server ...	// ... 在这里使用注入 的 DatabaseFixture	}	
}

Singleton Sample

这里以一个对 asp.net core API 的测试为例

  1. 自定义 Fixture

/// <summary>	
/// Shared Context https://xunit.github.io/docs/shared-context.html	
/// </summary>	
public class APITestFixture : IDisposable	
{	private readonly IWebHost _server;	public IServiceProvider Services { get; }	public HttpClient Client { get; }	public APITestFixture()	{	var baseUrl = $"http://localhost:{GetRandomPort()}";	_server = WebHost.CreateDefaultBuilder()	.UseUrls(baseUrl)	.UseStartup<TestStartup>()	.Build();	_server.Start();	Services = _server.Services;	Client = new HttpClient(new WeihanLi.Common.Http.NoProxyHttpClientHandler())	{	BaseAddress = new Uri($"{baseUrl}")	};	// Add Api-Version Header	// Client.DefaultRequestHeaders.TryAddWithoutValidation("Api-Version", "1.2");	Initialize();	Console.WriteLine("test begin");	}	/// <summary>	/// TestDataInitialize	/// </summary>	private void Initialize()	{	}	public void Dispose()	{	using (var dbContext = Services.GetRequiredService<ReservationDbContext>())	{	if (dbContext.Database.IsInMemory())	{	dbContext.Database.EnsureDeleted();	}	}	Client.Dispose();	_server.Dispose();	Console.WriteLine("test end");	}	private static int GetRandomPort()	{	var random = new Random();	var randomPort = random.Next(10000, 65535);	while (IPGlobalProperties.GetIPGlobalProperties().GetActiveTcpListeners().Any(p => p.Port == randomPort))	{	randomPort = random.Next(10000, 65535);	}	return randomPort;	}	
}	
[CollectionDefinition("APITestCollection")]	
public class APITestCollection : ICollectionFixture<APITestFixture>	
{	
}
  1. 自定义Collection

[CollectionDefinition("TestCollection")]	
public class TestCollection : ICollectionFixture<TestStartupFixture>	
{	
}
  1. 自定义一个 TestBase

[Collection("APITestCollection")]	
public class ControllerTestBase	
{	protected HttpClient Client { get; }	protected IServiceProvider Services { get; }	public ControllerTestBase(APITestFixture fixture)	{	Client = fixture.Client;	Services = fixture.Services;	}	
}
  1. 需要依赖注入的Test类写法

public class NoticeControllerTest : ControllerTestBase	
{	public NoticeControllerTest(APITestFixture fixture) : base(fixture)	{	}	[Fact]	public async Task GetNoticeList()	{	using (var response = await Client.GetAsync("/api/notice"))	{	Assert.Equal(HttpStatusCode.OK, response.StatusCode);	var responseString = await response.Content.ReadAsStringAsync();	var result = JsonConvert.DeserializeObject<PagedListModel<Notice>>(responseString);	Assert.NotNull(result);	}	}	[Fact]	public async Task GetNoticeDetails()	{	var path = "test-notice";	using (var response = await Client.GetAsync($"/api/notice/{path}"))	{	Assert.Equal(HttpStatusCode.OK, response.StatusCode);	var responseString = await response.Content.ReadAsStringAsync();	var result = JsonConvert.DeserializeObject<Notice>(responseString);	Assert.NotNull(result);	Assert.Equal(path, result.NoticeCustomPath);	}	}	[Fact]	public async Task GetNoticeDetails_NotFound()	{	using (var response = await Client.GetAsync("/api/notice/test-notice1212"))	{	Assert.Equal(HttpStatusCode.NotFound, response.StatusCode);	}	}	
}

运行测试,查看我们的 APITestFixture 是不是只实例化了一次,查看输出日志:

640?wx_fmt=png

可以看到我们输出的日志只有一次,说明在整个测试过程中确实只实例化了一次,只会启动一个 web server,确实是单例的

Memo

微软推荐的是用 Microsoft.AspNetCore.Mvc.Testing 组件去测试 Controller,但是个人感觉不如自己直接去写web 服务去测试,如果没必要引入自己不熟悉的组件最好还是不要去引入新的东西,否则可能就真的是踩坑不止了。

Reference

  • https://xunit.github.io/docs/shared-context.html

  • https://github.com/WeihanLi/ActivityReservation/tree/dev/ActivityReservation.API.Test

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

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

相关文章

P3711 仓鼠的数学题(伯努利数)

P3711 仓鼠的数学题 有关伯努利数的知识可以看我的上一篇题解链接&#xff08;写的超详细&#xff09;。 F(x)∑k0nSk(x)ak原本定义的Sk(x)∑i0xik根据伯努利数的定义Sk′(x)∑i0x−1ik则我们求F(x)∑k0nSk′(x)ak,答案即为F(x1)考虑先求F(x)∑k0nak1k1∑i0kCk1iBixk−i1∑k0n…

程序员自家种水果,新鲜包邮配送!

点击上面“蓝字”关注我们&#xff01;上次猕猴桃的活动一经推出&#xff0c;得到了广大粉丝的支持&#xff0c;我感到十分欣慰&#xff0c;非常感谢大家对我的信任。好多小伙伴&#xff0c;买了一箱尝过后又下单了好几箱。事实证明&#xff0c;品质才是销量的最佳保证。有些粉…

实现一个简单的基于码云(Gitee) 的 Storage

实现一个简单的基于码云(Gitee) 的 StorageIntro上次在 asp.net core 从单机到集群 一文中提到存储还不支持分布式&#xff0c;并立了一个 flag基于 github 或者 开源中国的码云实现一个 storage于是这两天就来填坑了。。实现了一个简单的基于开源中国的码云的 storage准备工作…

Java多线程的4种实现方式

** Java多线程的4种实现方式 ** 1&#xff1a;继承Thread并重写run方法&#xff0c;并调用start方法 /*** Java实现多线程的方式1* 继承Thread类&#xff0c;重写run方法* author hongbo.zhao 2019年4月12日 上午7:12:35*/ class MyThread extends Thread {Overridepublic …

采蘑菇的克拉莉丝(树链剖分)

采蘑菇的克拉莉丝 一个有点意思的树链剖分的题。 题意&#xff1a; 一棵树&#xff0c;有两种操作&#xff1a; ①&#xff1a;在点vvv放xxx个蘑菇。 ②&#xff1a;将起点变为vvv。 每次计算收集所有蘑菇的代价。 收集蘑菇的代价为&#xff0c;起点到所在蘑菇的路径上的…

HDU 6428 Problem C. Calculate(积性函数)

Problem C. Calculate ϕϕ∗ϵϕ∗μ∗Iϕ(n)∑d∣n(ϕ∗μ)(d)设g(n)∑d∣n(ϕ∗μ)(d)∑i1A∑j1B∑k1Cϕ(gcd(i,j2,k3))∑i1A∑j1B∑k1C∑d∣i,d∣j2,d∣k3(ϕ∗μ)(d)∑d1A(ϕ∗μ)(d)∑i1A∑j1B∑k1C[d∣i,d∣j2,d∣k3]\phi \phi * \epsilon \phi * \mu * I\\ \phi(n) …

Java线程的6种状态

线程的概念&#xff0c;以及线程的创建方式&#xff0c;见我之前写的博文 本篇文章主要讲Java线程的6种状态 6种状态&#xff1a;初始状态&#xff08;new&#xff09; 、可运行状态&#xff08;Runnable&#xff09;、运行状态&#xff08;Running&#xff09;、阻塞状态&am…

C. Goodbye Souvenir(CDQ 或 树套树)

C. Goodbye Souvenir ∑iLRi−preAi[preAi≥L]\sum\limits_{i L} ^{R} i - pre_{A_i} [pre_{A_i} \geq L]iL∑R​i−preAi​​[preAi​​≥L]&#xff0c;进一步考虑即∑i−preAi[i≤R,preAi≥L]\sum i - pre_{A_i}[i \leq R, pre_{A_i} \geq L]∑i−preAi​​[i≤R,preAi​​…

.NET Core 微信小程序支付——(统一下单)

最近公司研发了几个电商小程序&#xff0c;还有一个核心的电商直播&#xff0c;只要是电商一般都会涉及到交易信息&#xff0c;离不开支付系统&#xff0c;这里我们统一实现小程序的支付流程&#xff08;与服务号实现步骤一样&#xff09;。目录1、开通小程序的支付能力2、商户…

P4768 [NOI2018] 归程(kruskal 重构树)

P4768 [NOI2018] 归程 给定一个nnn个点&#xff0c;mmm条边的无向联通图&#xff0c;边的描述为[u,v,l,a][u, v, l, a][u,v,l,a]&#xff0c;表示uuu&#xff0c;vvv连有一条长度为lll&#xff0c;海拔为aaa的边&#xff0c; 有QQQ个询问&#xff0c;每次给出一个出发点uuu和…

用.NET写“算命”程序

前言“算命”&#xff0c;是一种迷信&#xff0c;我父亲那一辈却执迷不悟&#xff0c;有时深陷其中&#xff0c;有时为求一“上上签”&#xff0c;甚至不惜重金&#xff0c;向“天神”保佑。我曾看到过有些算命网站&#xff0c;可以根据人的生辰八字&#xff0c;来求得这个人一…

Java线程调度

线程调度指的是系统为线程分配CPU使用权的方式。主要有协同式线程调度和抢占式线程调度。 协同式线程调度&#xff08;Cooperative Threads-Scheduling&#xff09; 在多线程系统中&#xff0c;线程的执行时间由线程自身控制&#xff0c;执行结束后要主动通知系统切换到另一线…

#3551. [ONTAK2010]Peaks加强版(kruskal 重构树 + 主席树)

#3551. [ONTAK2010]Peaks加强版 我们要求从一个点出发经过困难值小于等于xxx的路径所能到达的山峰中第kkk高的是什么。 考虑按照边权升序&#xff0c;建议kruskalkruskalkruskal重构树&#xff0c;然后倍增向上跳&#xff0c;找到困难值小于等于xxx的深度最小的节点uuu&#…

ASP.NET Core 3.0 迁移避坑指南

一.前言.NET Core 3.0将会在 .NET Conf 大会上正式发布&#xff0c;截止今日发布了9个预览版&#xff0c;改动也是不少&#xff0c;由于没有持续关注&#xff0c;今天将前面开源的动态WebApi项目迁移到.NET Core 3.0还花了不少时间踩坑&#xff0c;给大家分享一下我在迁移过程中…

kruskal 重构树(讲解 + 例题)

kruskal重构树 如何建树 模仿kruskalkruskalkruskal&#xff0c;先将所有边排序。 依次遍历每一条边&#xff0c;如果这条边的两个节点&#xff08;u,vu, vu,v&#xff09;不在同一个连通块里面&#xff0c; 则新建一个nodenodenode节点&#xff0c;更新fa[u]fa[v]nodefa[u…

打不死我的,终将使我强大!DevOps黑客马拉松参赛心得

&#xff08;IDCF DevOps黑客马拉松到底是个啥活动&#xff1f;&#xff09;长得丑活得久、长得帅也惹人爱&#xff01;大家好&#xff0c;我是刘威。隆正信息的业务架构师-花名逸云。非常荣幸可以参加在北京举办的第一届DevOps黑客马拉松比赛。黑客马拉松不是突然冒出来的&…

Java线程池面试题

1、什么是线程池 java.util.concurrent.Executors提供了一个 java.util.concurrent.Executor接口的实现用于创建线程池 多线程技术主要解决处理器单元内多个线程执行的问题&#xff0c;它可以显著减少处理器单元的闲置时间&#xff0c;增加处理器单元的吞吐能力。 假设一个服…

F. Cheap Robot(kruskal 重构树)

F. Cheap Robot 给定一个无向连通图&#xff0c;每条边有边权&#xff0c;路过需要消耗对应的电量&#xff08;边权&#xff09;&#xff0c;有kkk个中心点&#xff0c; 问从a−>ba-> ba−>b&#xff0c;我们最少需要带多少电&#xff0c;设最小为CCC&#xff0c;当…

「标签管理」使用标签管理有道云笔记资料

因着大家对文件标签化比较高难道&#xff0c;需要熟悉一个标签工具软件&#xff0c;所以今天暂时来介绍个简单一些的网络资料的标签化管理&#xff0c;使用有道云笔记作为落地工具&#xff0c;同理在OneNote、印象笔记上原理类似。有道云笔记免费功能够用为了选择哪个笔记产品&…

JDK线程池实现

参见&#xff1a; https://www.cnblogs.com/trust-freedom/p/6594270.html http://ifeve.com/java-threadpool/ https://blog.csdn.net/u010963948/article/details/80573898 https://blog.csdn.net/moakun/article/details/80606029 https://www.cnblogs.com/liuzhihu/p/81773…