0

0

c# 使用 WebApplicationFactory 进行 ASP.NET Core 并发集成测试

幻夢星雲

幻夢星雲

发布时间:2026-01-18 05:39:08

|

676人浏览过

|

来源于php中文网

原创

WebApplicationFactory 默认不支持并发测试,需为每个任务调用 CreateClient() 获取独立 HttpClient;避免复用客户端、检查单例服务线程安全、使用唯一内存数据库名。

c# 使用 webapplicationfactory 进行 asp.net core 并发集成测试

WebApplicationFactory 默认不支持并发测试

直接在多个线程或 Task 中复用同一个 WebApplicationFactory 实例发起 HTTP 请求,大概率会遇到连接复用冲突、HTTP/2 流错乱、或 HttpRequestException: Error while copying content to a stream。这是因为 WebApplicationFactory 内部的 TestServer 虽然线程安全,但其默认配置的 HttpClient 实例(通过 CreateClient() 返回)**共享连接池且未隔离请求上下文**,尤其在高并发短时密集调用下容易触发底层 HttpHandler 状态竞争。

必须为每个并发测试任务创建独立 HttpClient 实例

不要复用 CreateClient() 返回的单个 HttpClient 对象。每次并发请求前都应调用一次 CreateClient(),让 WebApplicationFactory 为你生成带独立连接管理的新客户端。

  • CreateClient() 是轻量操作,不重建应用或服务器,只新建 HttpClient 和关联的 TestServer 连接句柄
  • 若手动复用 HttpClient,需确保它被正确配置为支持并发(例如设置 MaxConnectionsPerServer),但不如直接用 CreateClient() 简洁可靠
  • 在 xUnit 的 [Theory] + ParallelizableTask.WhenAll 场景中,务必在每个 Task 内部调用 CreateClient()
var factory = new WebApplicationFactory();
await Task.WhenAll(Enumerable.Range(0, 10).Select(async _ =>
{
    // ✅ 正确:每个任务独立获取 client
    using var client = factory.CreateClient();
    var response = await client.GetAsync("/api/values");
    response.EnsureSuccessStatusCode();
}));

注意 TestServer 的同步上下文与异步陷阱

TestServer 内部使用同步上下文模拟 ASP.NET Core 的请求生命周期,但在高并发下若测试代码意外捕获并重用 SynchronizationContext(比如在 UI 测试项目中引入了 WinForms/WPF 引用),可能引发死锁或响应延迟。xUnit 默认无同步上下文,但某些自定义测试基类或第三方断言库可能干扰。

ListenHub
ListenHub

超真实的AI播客生成器

下载
  • 确认测试项目未引用 System.Windows.FormsPresentationCore
  • 避免在测试方法中调用 Task.Wait()Result —— 必须用 await
  • 若需调试,可在 WebApplicationFactory 派生类中重写 CreateWebHostBuilder() 并添加日志中间件,观察并发请求是否被串行化处理

并发测试失败时优先检查依赖服务状态

WebApplicationFactory 启动的是完整应用,包括所有注册的单例服务(如 IMemoryCache、数据库上下文、事件总线等)。并发请求共享这些实例,若服务内部状态非线程安全(例如手动实现的静态计数器、未加锁的字典写入),会导致间歇性失败。

  • 检查 Program.cs 中是否将有状态对象注册为 AddSingleton 却未做同步保护
  • 对集成测试中的数据库操作,建议使用临时内存数据库(AddDbContext(o => o.UseInMemoryDatabase(...))),并确保每个测试用唯一数据库名(如 Guid.NewGuid().ToString()
  • 若使用真实数据库或外部服务(Redis、RabbitMQ),并发测试前需确认其连接池配置足够,否则瓶颈不在应用层而在下游

真正难排查的并发问题往往不出在 WebApplicationFactory 本身,而在于你没意识到某个 Singleton 服务正被十个请求同时修改同一块内存。

相关专题

更多
rabbitmq和kafka有什么区别
rabbitmq和kafka有什么区别

rabbitmq和kafka的区别:1、语言与平台;2、消息传递模型;3、可靠性;4、性能与吞吐量;5、集群与负载均衡;6、消费模型;7、用途与场景;8、社区与生态系统;9、监控与管理;10、其他特性。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

201

2024.02.23

什么是中间件
什么是中间件

中间件是一种软件组件,充当不兼容组件之间的桥梁,提供额外服务,例如集成异构系统、提供常用服务、提高应用程序性能,以及简化应用程序开发。想了解更多中间件的相关内容,可以阅读本专题下面的文章。

178

2024.05.11

Golang 中间件开发与微服务架构
Golang 中间件开发与微服务架构

本专题系统讲解 Golang 在微服务架构中的中间件开发,包括日志处理、限流与熔断、认证与授权、服务监控、API 网关设计等常见中间件功能的实现。通过实战项目,帮助开发者理解如何使用 Go 编写高效、可扩展的中间件组件,并在微服务环境中进行灵活部署与管理。

212

2025.12.18

while的用法
while的用法

while的用法是“while 条件: 代码块”,条件是一个表达式,当条件为真时,执行代码块,然后再次判断条件是否为真,如果为真则继续执行代码块,直到条件为假为止。本专题为大家提供while相关的文章、下载、课程内容,供大家免费下载体验。

89

2023.09.25

scripterror怎么解决
scripterror怎么解决

scripterror的解决办法有检查语法、文件路径、检查网络连接、浏览器兼容性、使用try-catch语句、使用开发者工具进行调试、更新浏览器和JavaScript库或寻求专业帮助等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

187

2023.10.18

500error怎么解决
500error怎么解决

500error的解决办法有检查服务器日志、检查代码、检查服务器配置、更新软件版本、重新启动服务、调试代码和寻求帮助等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

280

2023.10.25

线程和进程的区别
线程和进程的区别

线程和进程的区别:线程是进程的一部分,用于实现并发和并行操作,而线程共享进程的资源,通信更方便快捷,切换开销较小。本专题为大家提供线程和进程区别相关的各种文章、以及下载和课程。

481

2023.08.10

windows查看端口占用情况
windows查看端口占用情况

Windows端口可以认为是计算机与外界通讯交流的出入口。逻辑意义上的端口一般是指TCP/IP协议中的端口,端口号的范围从0到65535,比如用于浏览网页服务的80端口,用于FTP服务的21端口等等。怎么查看windows端口占用情况呢?php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

578

2023.07.26

高德地图升级方法汇总
高德地图升级方法汇总

本专题整合了高德地图升级相关教程,阅读专题下面的文章了解更多详细内容。

43

2026.01.16

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
进程与SOCKET
进程与SOCKET

共6课时 | 0.3万人学习

Redis+MySQL数据库面试教程
Redis+MySQL数据库面试教程

共72课时 | 6.4万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号