我們希望將一個項目(dll)看做一個模塊/插件,一個模塊往往需要在應用啟動時做一些初始化工作,比如向IOC容器添加一些服務,為應用配置對象添加自己的數據源;也希望在應用關閉時做一些收尾工作,asp.net core為我們提供了這種機制,先來看看如何使用,再講講原理。
如何使用?
1、創建asp.net core 3.1的web應用程序,WebApplication6
2、創建我們的模塊/插件項目,一個Standard2.1項目叫ClassLibrary2
3、在插件項目ClassLibrary2中定義實現IHostingStartup的類
1 public class HostingStartup : IHostingStartup 2 { 3 public void Configure(IWebHostBuilder builder) 4 { 5 //向IOC容器添加或替換各種服務 6 builder.ConfigureServices((c,b)=> { 7 b.AddSingleton<Class1>(); 8 }); 9 //為應用配置對象添加更多數據源 10 builder.ConfigureAppConfiguration(c => { 11 c.AddInMemoryCollection(new Dictionary<string, string> { {"a","tttt" } }); 12 });
//處理當前模塊的其它初始化操作 13 } 14 }
5、在插件項目ClassLibrary2中隨便找個類文件中,設置[assembly: HostingStartupAttribute(typeof(ClassLibrary2.HostingStartup))]
6、在主程序WebApplication6中設置環境變量,
除了這樣配置,我們也可以在主程序的Program.main配置主機時手動覆蓋配置值,以達到設置插件關聯的程序集的目的,多個插件程序集用分號“;”分割
1 public static IHostBuilder CreateHostBuilder(string[] args) => 2 Host.CreateDefaultBuilder(args) 3 .ConfigureWebHostDefaults(webBuilder =>{ 4 webBuilder.UseSetting(WebHostDefaults.HostingStartupAssembliesKey, "ClassLibrary2"); 5 webBuilder.UseStartup<Startup>(); 6 });
還可以使用webBuilder.UseSetting(WebHostDefaults.HostingStartupExcludeAssembliesKey, "ClassLibrary3");排出一些程序集,不把這些程序集當做插件來加載
原理是啥?
程序啟動時會根據環境變量找到對應的插件程序集
根據程序集找到關聯的 HostingStartupAttribute
通過 HostingStartupAttribute拿到插件啟動類並調用其Confiure方法
方法內部可以做此插件的初始化工作、向主機IOC容器注冊各種服務、設置應用配置的數據源等
多個模塊的啟動順序是啥?
按配置的順序加載模塊的,所以最少依賴的模塊應該寫在前面,這個設計不如abp
模塊之間如何通訊?
辦法1、直接引用,配置時最好將被依賴的模塊放前面
辦法2、不添加直接引用關系,而用中間層實現
如添加一個中間項目,定義各種接口,由模塊B來實現,在模塊B中向容器注冊自己的服務。模塊A引用中間類庫,直接在使用地方注入接口就ok啦
應用關閉時模塊如果做一些收尾工作?
可以定義一個應用生命周期事件處理程序(實現IHostedApplicationLifetime),在不同事件中定義此模塊的收尾工作。然后在模塊啟動類中向IOC注冊這個服務。但這樣有個問題,默認的生命周期事件處理程序被我們替換掉了,所以我們的類應該用構造函數注入IHostedApplicationLifetime,然后再調用它一次
每次寫點東西都語無倫次,哈哈