当前位置: 首页 > java >正文

单测时如何让 mock 的接口在长链路调用时一直生效

问题描述

在使用@MockBean 和 @SpyBean 注解的时候,需要注意:@MockBean 会完全 mock 掉这个 bean,也就是说,假如你指定了 在调用方法 A 时按 mock 期望返回,而没有指定调用方法 B 时返回什么,那么你在调用方法 B 时,方法 B 会直接返回 Null,换句话说,当你使用 @MockBean 的时候,那个 bean 除了按你给出的 case 返回之外,其余情况都返回 null。而@SpyBean 不一样,@SpyBean 支持“非预设时进行真实调用”,即假如你指定了 在调用方法 A 时按 mock 期望返回,而没有指定调用方法 B 时返回什么,那么你在调用方法 B 时,方法 B 会进行真实的调用,这是 @SpyBean 比 @MockBean 要丰富的功能。但是,@SpyBean 只能用在真实的 bean 身上,而用在接口上时,则会直接报错,错误内容是接口不能够被实例化。因此在对外部接口进行 mock 时,往往只能使用@MockBean。

但是,@MockBean 在用在接口上时,有时会出现一种 case 是:@MockBean mock 的 bean 并不能在长链路调用时一直生效。比如,现有的调用链路是 A->B->C->D,D 是外部接口,现在我使用 @MockBean 标记 D,期望 D 按照我 mock 的行为来返回,但是会出现一个问题:在单测中,我的入口是 A,结果走到 D 时仍旧会发起真实调用,而如果你再写一个单测,直接测调用 D 时的表现,却发现 D 的表现符合当初的 mock 预期,这是为什么呢?

举例说明:

@RunWith(SpringJUnit4ClassRunner.class)
@SpringBootTest(classes = BgMoriaApplication.class)
@ActiveProfiles("test")
@WebAppConfiguration
@Slf4j
@DirtiesContext(classMode = DirtiesContext.ClassMode.AFTER_EACH_TEST_METHOD)
@ComponentScan(excludeFilters = @ComponentScan.Filter(type = FilterType.ASSIGNABLE_TYPE,classes = {TigerConcurrentConsumer.class, TigerMessageProducer.class}
))
public abstract class BaseApiTest {@MockBeanprivate RegulatedUserAddressSnapshotQueryForSPService mockServicepublic void mockRegulatedQueryAddressEmptyResponse() {Mockito.when(mockService.queryAddressInfoForPaymentScene(any())).thenReturn(buildEmptyBaseResponse(QueryAddressSnapshotResponse.class));}
}

在上面的单测基础类中,我定义了要 mock 的接口为mockService.queryAddressInfoForPaymentScene,并期望他返回一个空对象。

而我实际的调用链路为:

  1.addressSnapshotService.queryHolderNameAddress   ->2.raphaelIntegration.queryAddressSnapshot  ->3. regulatedUserAddressSnapshotQueryForSPService.queryAddressInfoForPaymentScene

我的单测为:

@Slf4j
public class SorosRiskSpiTest extends BaseApiTest{@Testpublic void testQueryRegulatedHolderNameAddressAndMobileWithEmptyResponse_NonRegulatedNotNullResponse() {mockRegulatedQueryAddressEmptyResponse();mockRegulatedQueryAddressMobileEmptyResponse();Long uid = 27953383161566L;String addressSnapshotId = "228100002134397257";String regionId = "128";UserAddressSnapshotVO addressSnapshotVO = addressSnapshotService.queryHolderNameAddress(uid, addressSnapshotId, regionId);Assert.notNull(addressSnapshotVO);AddressSnapshotMobileVO userAddressSnapshotMobileVO = addressSnapshotService.queryAddressSnapshotMobile(uid, addressSnapshotId, regionId);Assert.notNull(userAddressSnapshotMobileVO);}
}

运行单测会发现,还是真实地调用了regulatedUserAddressSnapshotQueryForSPService.queryAddressInfoForPaymentScene

可能的一个原因是Bean 依赖注入的顺序 和 Spring 上下文刷新机制有关,也可能跟多层代理机制有关。RaphaelIntegration 在 Spring 容器初始化时,已经注入了原始的 RegulatedUserAddressSnapshotQueryForSPService Bean,即使后续通过 @MockBean 创建了 Mock 对象,RaphaelIntegration 内部的引用仍指向原始 Bean;而对于 dubbo 接口(而 Dubbo 接口的实现原理就是动态代理),Spring 可能为其创建了动态代理,导致 @MockBean 无法直接覆盖。

解决办法

怎么才能让这个 mock 的 bean 在长链路中一直生效呢?用下面的写法:

@RunWith(SpringJUnit4ClassRunner.class)
@SpringBootTest(classes = BgMoriaApplication.class)
@ActiveProfiles("test")
@WebAppConfiguration
@Slf4j
@DirtiesContext(classMode = DirtiesContext.ClassMode.AFTER_EACH_TEST_METHOD)
@ComponentScan(excludeFilters = @ComponentScan.Filter(type = FilterType.ASSIGNABLE_TYPE,classes = {TigerConcurrentConsumer.class, TigerMessageProducer.class}
))
public abstract class BaseApiTest {@Autowiredprivate ApplicationContext ctx;@MockBeanprivate RegulatedUserAddressSnapshotQueryForSPService mockServicepublic void mockRegulatedQueryAddressEmptyResponse() {Mockito.when(mockService.queryAddressInfoForPaymentScene(any())).thenReturn(buildEmptyBaseResponse(QueryAddressSnapshotResponse.class));ReflectionTestUtils.setField(ctx.getBean(RaphaelIntegration.class),"regulatedUserAddressSnapshotQueryForSPService",mockService);}
}

上面的写法实际上就是用 mock 的 bean 强制覆盖上下文中的 bean,这样你在调用RaphaelIntegration 时,里面的regulatedUserAddressSnapshotQueryForSPService 就是 mock 的 bean。

进阶

那如果对于同一个接口 service 下面的多个接口,我想让 A 按自己 mock 的结果返回,而让 B 进行真实调用,这又如何实现呢?

@RunWith(SpringJUnit4ClassRunner.class)
@SpringBootTest(classes = BgMoriaApplication.class)
@ActiveProfiles("test")
@WebAppConfiguration
@Slf4j
@DirtiesContext(classMode = DirtiesContext.ClassMode.AFTER_EACH_TEST_METHOD)
@ComponentScan(excludeFilters = @ComponentScan.Filter(type = FilterType.ASSIGNABLE_TYPE,classes = {TigerConcurrentConsumer.class, TigerMessageProducer.class}
))
public abstract class BaseApiTest {@Resource@Qualifier("regulatedUserAddressSnapshotQueryForSPService")private RegulatedUserAddressSnapshotQueryForSPService realService;private RegulatedUserAddressSnapshotQueryForSPService mockService = Mockito.mock(RegulatedUserAddressSnapshotQueryForSPService.class);;@Autowiredprivate ApplicationContext ctx;public void mockRegulatedQueryAddressMobileNullResponse() {Mockito.when(mockService.queryAddressRealMobileForPaymentScene(any())).thenReturn(null);ReflectionTestUtils.setField(ctx.getBean(RaphaelIntegration.class),"regulatedUserAddressSnapshotQueryForSPService",mockService);}public void mockRegulatedQueryAddressRealResponse() {Mockito.when(mockService.queryAddressInfoForPaymentScene(any())).thenAnswer(inv -> realService.queryAddressInfoForPaymentScene(inv.getArgument(0)));ReflectionTestUtils.setField(ctx.getBean(RaphaelIntegration.class),"regulatedUserAddressSnapshotQueryForSPService",mockService);}
}

如上所示,通过 thenAnswer 的方式来动态路由 mock 的路线,从而实现同一个 service 下的不同接口有不同的表现。

http://www.xdnf.cn/news/14222.html

相关文章:

  • 从STM32到NXP:GPIO就像装修房子,多了个“智能开关”
  • 基于 SpringBoot+Servlet+JSP 的医院医保管理系统的设计与实现,论文7000字,可根据实际情况调整
  • ES+索引库文档操作
  • [CVPR 2025] DiCo:动态协作网络助力半监督3D血管分割新突破
  • AI Agent实战 - LangChain+Playwright构建火车票查询Agent
  • 人工智能学习28-BP过拟合
  • [k8s]--exec探针详细解析
  • java常见第三方依赖以及相关安全问题
  • http1.x VS http2.x 协议
  • Spring Cloud Alibaba 中间件
  • 硬编码(修改RIP相关指令)
  • HTML+CSS 半透明登录框
  • (LeetCode每日一题) 2566. 替换一个数字后的最大差值 ( 贪心 )
  • 安防市场的中小企业突围——从竞品分析到破局路径的思考
  • Spring Boot中Controller层规划与最佳实践详解
  • 【北京迅为】iTOP-4412精英版使用手册-第二十一章 延时函数专题
  • Python爬虫-批量爬取快手视频并将视频下载保存到本地
  • BeckHoff PLC --> 料筐(KLT Box)自动对中与抓取程序分析
  • Deep Research Agent的深度与广度如何保证
  • OSGI 是什么,有哪些具体应用、java8、9、10、11比较
  • C++操作系统与网络编程(针对特定岗位)
  • SpringBoot打包运行原理和加载机制原理
  • 从大数据到大模型:我们是否在重蹈覆覆辙
  • 一文详解前缀和:从一维到二维的高效算法应用
  • Java相关-链表-设计链表-力扣707
  • JS进阶 Day02
  • 在tensorrt engine中提高推理性能小记
  • 互联网大厂Java求职面试:云原生架构与微服务设计中的复杂挑战
  • Flask文件上传与异常处理完全指南
  • create_react_agent + MCP tools