更優雅的在 Xunit 中使用依賴註入

来源:https://www.cnblogs.com/weihanli/archive/2020/07/16/13324503.html
-Advertisement-
Play Games

Xunit.DependencyInjection 7.0 發佈了 Intro 上次我們已經介紹過一次大師的 Xunit.DependencyInjection https://www.cnblogs.com/weihanli/p/xuint-dependency-injection.html ,最 ...


Xunit.DependencyInjection 7.0 發佈了

Intro

上次我們已經介紹過一次大師的 Xunit.DependencyInjection https://www.cnblogs.com/weihanli/p/xuint-dependency-injection.html ,最近大師完成了 7.0 的重構並且已經正式發佈,已經可以直接安裝使用了

7.0 為我們帶來了更好的編程體驗,在 6.x 的版本中,我們的 Startup 需要繼承於 DependencyInjectionTestFramework 而且需要設置一個 assembly attribute,這在 7.0 中都不需要了,下麵我們來看看有了哪些變化

Startup 的變化

首先來看大師給出的 diff


-[assembly: TestFramework("Your.Test.Project.Startup", "Your.Test.Project")]

namespace Your.Test.Project
{
-   public class Startup : DependencyInjectionTestFramework
+   public class Startup
    {
-       public Startup(IMessageSink messageSink) : base(messageSink) { }

-       protected void ConfigureServices(IServiceCollection services)
+       public void ConfigureServices(IServiceCollection services)
        {
            services.AddTransient<IDependency, DependencyClass>();
        }

-       protected override IHostBuilder CreateHostBuilder() =>
-           base.CreateHostBuilder(assemblyName)
-               .ConfigureServices(ConfigureServices);

-       protected override void Configure(IServiceProvider provider)
+       public void Configure(IServiceProvider provider)
    }
}
  1. 移除了 TestFramework assembly attribute
  2. 不再需要繼承於 DependencyInjectionTestFramework
  3. 也因為上面的不需要繼承,所以原本要 override 的方法可以不 override 了,原來是 protected 的方法現在需要改成 public

這樣改了之後首先我們在使用的時候無需知道 DependencyInjectionTestFramework 的存在了,而且可以更符合 asp.net core Startup 的使用習慣,可以屏蔽掉很多實現細節,用戶只需要在 Startup 註冊自己的邏輯即可,更為專註於自己的邏輯而無需關心框架所做的事情

新的 Startup 解析

我把上一篇文章寫的示例用升級到了新的版本,下麵是更新後的示例代碼

namespace XUnitDependencyInjectionSample
{
    public class Startup
    {
        // 自定義 HostBuilder ,可以沒有這個方法,沒有這個方法會使用預設的 hostBuilder,通常直接使用 `ConfigureHost` 應該就夠用了
        // public IHostBuilder CreateHostBuilder()
        // {
        //     return new HostBuilder()
        //         .ConfigureAppConfiguration(builder =>
        //         {
        //             // 註冊配置
        //             builder
        //                 .AddInMemoryCollection(new Dictionary<string, string>()
        //                 {
        //                     {"UserName", "Alice"}
        //                 })
        //                 .AddJsonFile("appsettings.json")
        //                 ;
        //         })
        //         .ConfigureServices((context, services) =>
        //         {
        //             // 註冊自定義服務
        //             services.AddSingleton<IIdGenerator, GuidIdGenerator>();
        //             if (context.Configuration.GetAppSetting<bool>("XxxEnabled"))
        //             {
        //                 services.AddSingleton<IUserIdProvider, EnvironmentUserIdProvider>();
        //             }
        //         })
        //         ;
        // }

        // 自定義 host 構建
        public void ConfigureHost(IHostBuilder hostBuilder)
        {
            hostBuilder
                .ConfigureAppConfiguration(builder =>
                {
                    // 註冊配置
                    builder
                        .AddInMemoryCollection(new Dictionary<string, string>()
                        {
                            {"UserName", "Alice"}
                        })
                        .AddJsonFile("appsettings.json")
                        ;
                })
                .ConfigureServices((context, services) =>
                {
                    // 註冊自定義服務
                    services.AddSingleton<IIdGenerator, GuidIdGenerator>();
                    if (context.Configuration.GetAppSetting<bool>("XxxEnabled"))
                    {
                        services.AddSingleton<IUserIdProvider, EnvironmentUserIdProvider>();
                    }
                })
                ;
        }

        // 支持的形式:
        // ConfigureServices(IServiceCollection services)
        // ConfigureServices(IServiceCollection services, HostBuilderContext hostBuilderContext)
        // ConfigureServices(HostBuilderContext hostBuilderContext, IServiceCollection services)
        public void ConfigureServices(IServiceCollection services, HostBuilderContext hostBuilderContext)
        {
            services.TryAddSingleton<CustomService>();
        }

        // 可以添加要用到的方法參數,會自動從註冊的服務中獲取服務實例,類似於 asp.net core 里 Configure 方法
        public void Configure(IServiceProvider applicationServices, IIdGenerator idGenerator)
        {
            // 有一些測試數據要初始化可以放在這裡
            // InitData();
        }
    }
}

在新的版本中 Startup 和 asp.net core 里的 Startup 更加相像了,

會多一個 CreateHostBuilder/ConfigureHost(IHostBuilder) 的方法,允許用戶自定義 Host 的構建,也可以沒有這個方法

ConfigureServices 方法允許用戶增加 HostBuilderContext 作為參數,可以通過 hostBuilderContext 來獲取配置信息,也可以在 CreateHostBuilder/ConfigureHost(IHostBuilder) 里註冊也是一樣的

註冊配置/服務和 asp.net core 里一模一樣,有數據或配置需要在項目啟動時初始化的,可以放在 Configure 方法做,有點類似於 asp.net core 里 Startup 中的 Configure 方法,可以將需要的服務作為方法參數,執行時會自動從註冊的服務中獲取

Startup 的尋找方法

預設的 Startup 通常是 ProjectName.Startup,通常在項目根目錄下創建一個 Startup 是不需要配置的,如果不是或不起作用,可以參考下麵 Startup 的尋找規則

如果要使用一個特別的 Startup, 你可以通過在項目文件的 PropertyGroup 部分定義 XunitStartupAssemblyXunitStartupFullName,具體規則如下

<Project>
  <PropertyGroup>
    <XunitStartupAssembly>Abc</XunitStartupAssembly>
    <XunitStartupFullName>Xyz</XunitStartupFullName>
  </PropertyGroup>
</Project>
XunitStartupAssembly XunitStartupFullName Startup
Your.Test.Project.Startup, Your.Test.Project
Abc Abc.Startup, Abc
Xyz Xyz, Your.Test.Project
Abc Xyz Xyz, Abc

More

除了上面的 Startup 的改動之外,新版本還支持了 xunit 中 fixture 的依賴註入,似乎是由一個外國小哥提的 PR, 詳見:https://github.com/pengweiqhca/Xunit.DependencyInjection/pull/21

有了這個神器,在測試代碼中使用依賴註入要方便很多了,還沒有用起來的可以準備上手了~~

Reference


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • function afterCreate(sender){ Ext.util.CSS.createStyleSheet( "#"+sender.id+"{overflow:hidden; position:relative; }" +"#"+sender.id+" .hd{ height:15px; ...
  • 前言 本文的文字及圖片來源於網路,僅供學習、交流使用,不具有任何商業用途,版權歸原作者所有,如有問題請及時聯繫我們以作處理。 項目實現知識點: Pandas庫及pyecharts庫 Pandas:數據分析和處理工具。 pd.read_csv():讀取csv文件。 pyecharts:繪圖庫,提供30 ...
  • 作者: zyl910 一、緣由 有些時候需要替換zip內的文件。 網上的辦法大多是——先解壓,然後對解壓目錄替換文件,最後再重新壓縮。該辦法需要比較繁瑣,且需要一個臨時目錄。 於是想找無需解壓的方案。 後來找到利用 ZipInputStream、ZipOutputStream 實現該功能的辦法。 二 ...
  • 1.線程狀態 新生狀態->就緒狀態->運行狀態->死亡狀態 註: 1.線程死亡後不能重新開啟 2.線程阻塞後進入就緒狀態 註: 1.每個線程new一個對象後,都有自己的工作空間,一個工作空間匹配一個線程 2.進入就緒狀態的四種方法:start方法,阻塞接觸,yield方法,jvm將cpu從本地線程切 ...
  • Java File類基礎解析 使用遞歸來遍歷目錄的代碼 2 package File; import java.io.File; public class Main { public static void main(String[] args) { //要遍歷的文件夾的根目錄 String roo ...
  • Java File類基礎解析 1 File類的構造方法 public File(String pathname) :通過給定的路徑名字元轉換為抽象路徑名來創建新的File實例 String path = new String("D:\\a.text"); File file = new File(p ...
  • 不知不覺,2020年已經過去一半了,最近突然反應過來自己也看了不少文獻資料了,就想著把看過的文獻和覺得比較好的書籍做一個總結,基本都是大數據分散式領域的,回顧自己學識的同時,也給想從事或這個領域的小伙伴一些參考 ...
  • 前言 本文的文字及圖片來源於網路,僅供學習、交流使用,不具有任何商業用途,版權歸原作者所有,如有問題請及時聯繫我們以作處理。 作者:Hanz — 1 — 如果你對本文的代碼感興趣,可以去 Github (文末提供)里查看。第一次運行的時候會報一個錯誤(還沒找到解決辦法),不過只要再運行一次就正常了。 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...