Autofac 通过 PreserveExistingDefaults 解决单元测试 Fake 对象被覆盖

2021-05-08 09:36:58 浏览数 (1)

在使用 Autofac 作为 IoC 容器,因为 Autofac 默认的创建时机是在主机运行时。而在此 Module 被 Load 时注入的对象的注入的时机,将会在单元测试 Fake 注入之后,这就意味着 Load 时注入的对象将会覆盖 Fake 的对象。可以通过调用 Autofac 的 PreserveExistingDefaults 方法解决覆盖的问题

在进行集成测试,需要注入一些 Fake 的或者 Mock 的等用来测试的对象,这些对象期望替换掉原有的业务逻辑的对象。而在使用 Autofac 框架,将因为对象创建时机的问题,而让单元测试不好玩

单元测试注入的顺序,是在业务对象注入之前,因此业务对象将会替换掉单元测试注入的对象

通过 PreserveExistingDefaults 方法,可以在框架判断,如果在此之前已有注册,那么将不再进行注册,代码如下

代码语言:javascript复制
            builder.RegisterType<Foo>().As<IFoo>()
                // 通过 PreserveExistingDefaults 可以在已经注册过了的应用,不会被覆盖为 Foo 类型
                // 在单元测试使用,单元测试注入了测试用的消费者,可以不被覆盖
                .PreserveExistingDefaults();

此时就可以在单元测试中,通过如下代码注入 FakeFoo 对象

代码语言:javascript复制
            Microsoft.Extensions.Hosting.Host.CreateDefaultBuilder()
                .ConfigureWebHostDefaults(webBuilder =>
                {
                    webBuilder.UseStartup<Startup>();
                    webBuilder.UseTestServer(); //关键是多了这一行建立TestServer
                })
                // 使用 auto fac 代替默认的 IOC 容器 
                .UseServiceProviderFactory(new AutofacServiceProviderFactory(builder =>
                {
                    builder.RegisterModule(new FakeFooModule());
                }))

    class FakeFooModule : Autofac.Module
    {
        protected override void Load(ContainerBuilder builder)
        {
            builder.RegisterType<FakeFoo>().As<FakeFoo>().As<IFoo>().SingleInstance();
        }
    }

上面代码就是尝试注入 FakeFoo 作为 IFoo 服务,在业务逻辑里面,将判断 IFoo 服务是否已注册,如果没有被注册,才注册为 Foo 对象

更多集成测试请看 asp dotnet core 基于 TestServer 做集成测试


本文会经常更新,请阅读原文: https://blog.lindexi.com/post/Autofac-通过-PreserveExistingDefaults-解决单元测试-Fake-对象被覆盖.html ,以避免陈旧错误知识的误导,同时有更好的阅读体验。

本作品采用 知识共享署名-非商业性使用-相同方式共享 4.0 国际许可协议 进行许可。欢迎转载、使用、重新发布,但务必保留文章署名林德熙(包含链接: https://blog.lindexi.com ),不得用于商业目的,基于本文修改后的作品务必以相同的许可发布。如有任何疑问,请 与我联系 。

无盈利,不卖课,做纯粹的技术博客

0 人点赞