.Net Core中使用Autofac替換自帶的DI容器的示例
為什么叫淺談呢?就是字面上的意思,講得比較淺,又不是不能用(這樣是不對(duì)的)?。。?/p>
Aufofac大家都不陌生了,說(shuō)是.Net生態(tài)下最優(yōu)秀的IOC框架那是一點(diǎn)都過(guò)分。用的人多了,使用教程也十分豐富,官網(wǎng)教程也比較詳細(xì)(如果英文功底還不錯(cuò)的話)。
那我為什么還要寫這樣一篇博客呢,一是用作學(xué)習(xí)筆記,二就是閑的。
廢話不多說(shuō),開始正文
項(xiàng)目創(chuàng)建
云創(chuàng)建一個(gè).Net Core Api項(xiàng)目,然后再添加一個(gè)類庫(kù),大概就是下面這樣的結(jié)構(gòu):
新建一個(gè)類庫(kù)項(xiàng)目,分別添加一個(gè)接口文件與類文件:
就這樣,我們的演示方案就搭建完成了,下面就到了演示階段。
方案演示
原始方案
俗話說(shuō)的好,沒(méi)有對(duì)象 new 一個(gè)就對(duì)了:
[HttpGet] public string Original() { IUserService userService = new UserService(); return userService.GetName("Original"); }
結(jié)果當(dāng)然是沒(méi)問(wèn)題的:
.Net Core自帶DI
微軟給我們提供的 DI 解決方案。如果是小項(xiàng)目,需要注入的服務(wù)不多,簡(jiǎn)直無(wú)敵好用,缺點(diǎn)就是不能批量注入,下面我們來(lái)復(fù)習(xí)一下:
先在 Startup 里面的 ConfigureServices 方法內(nèi)注入(默認(rèn)且只能構(gòu)造函數(shù)注入)
services.AddScoped<IUserService, UserService>();
然后在控制器中拿到剛才注入的服務(wù):
public class DefaultController : ControllerBase { private readonly IUserService userService; public DefaultController(IUserService _userService) { this.userService = _userService; } [HttpGet] public string CoreDI() { return userService.GetName("CoreDI"); } }
很顯然,一點(diǎn)問(wèn)題都沒(méi)有:
Autofac
注意事項(xiàng)說(shuō)在前面:
在 .Net Core2 中一般是把Startup
的ConfigureServices
方法返回值類型改為IServiceProvider,然后通過(guò)構(gòu)建Autofac容器并注入服務(wù)后返回。
在 .Net Core3.0之后,集成方式做了部分調(diào)整
下面演示的版本是.Net Core 3.1,也就是調(diào)整后的版本。
1、先引用 Autofac 的包,看看這下載次數(shù)
2、在 Program 中改用 Autofac 來(lái)實(shí)現(xiàn)依賴注入
public static IHostBuilder CreateHostBuilder(string[] args) => Host.CreateDefaultBuilder(args) // 就是這句 .UseServiceProviderFactory(new AutofacServiceProviderFactory()) .ConfigureWebHostDefaults(webBuilder => { webBuilder.UseStartup<Startup>(); });
3、添加我們自定義的 Autofac 注冊(cè)類,并注冊(cè)我們需要的服務(wù)(默認(rèn)構(gòu)造函數(shù)注入,支持屬性注入)
public class AutofacModuleRegister : Autofac.Module { //重寫Autofac管道Load方法,在這里注冊(cè)注入 protected override void Load(ContainerBuilder builder) { builder.RegisterType<UserService>().As<IUserService>(); } }
4、在 Startup 類中添加方法:ConfigureContainer,
public void ConfigureContainer(ContainerBuilder builder) { // 直接用Autofac注冊(cè)我們自定義的 builder.RegisterModule(new AutofacModuleRegister()); }
5、大功告成,控制器內(nèi)的方法甚至不用去改
public class DefaultController : ControllerBase { private readonly IUserService userService; public DefaultController(IUserService _userService) { this.userService = _userService; } [HttpGet] public string Autofac() { return userService.GetName("Autofac"); } }
演示到這里就結(jié)束了,是不是感覺(jué) Autofac 比自帶的 DI 還要麻煩。其實(shí)不然,下面我們就來(lái)看看 Autofac 對(duì)比自帶 DI 的一些特有特性。
不同的特性
批量注入
之前的項(xiàng)目我們有了用戶 UserService,需求更新,加入了商品(ProductService),有了商品那又怎么能少得了訂單(OrderService),那后面是不是還得有售后、物流、倉(cāng)庫(kù)、營(yíng)銷......
如果是.Net Core 自帶的注入框架,那就只能不停的:
services.AddScoped<IProductService, ProductService>(); services.AddScoped<IOrderService, OrderService>(); ......
這時(shí)候,Autofac 的好處就體現(xiàn)出來(lái)了:批量注入。
我們先回到上面的:AutofacModuleRegister 類,加入下面這段代碼:
// 服務(wù)項(xiàng)目程序集 Assembly service = Assembly.Load("XXX.Service"); // 服務(wù)接口項(xiàng)目程序集 Assembly iservice = Assembly.Load("XXX.IService"); builder.RegisterAssemblyTypes(service, iservice) .Where(t => t.FullName.EndsWith("Service") && !t.IsAbstract) .InstancePerLifetimeScope() .AsImplementedInterfaces();
上面的代碼就是批量注入 XXX.Service 與 XXX.IService 項(xiàng)目下的服務(wù)與接口。
注意:如果需要注入的服務(wù)沒(méi)有 interfac ,那么builder.RegisterAssemblyTypes 就只需要傳一個(gè)程序集就OK了。如果服務(wù)與接口同在一個(gè)項(xiàng)目,那也是要傳兩個(gè)程序集的哦。
然后我們?cè)诳刂破魅ネㄟ^(guò)構(gòu)造函數(shù)獲取注入的實(shí)例:
private readonly IUserService userService; private readonly IProductService productService; public DefaultController(IUserService _userService, IProductService _productService) { this.userService = _userService; this.productService = _productService; }
再對(duì)之前的 Autofac 接口添油加醋:
[HttpGet] public string Autofac() { var name = userService.GetName("Autofac"); return productService.Buy(name, "批量注入"); }
結(jié)果自然是沒(méi)有問(wèn)題的,如果后續(xù)需要加入其它服務(wù)都不用再單獨(dú)注入了,是不是優(yōu)點(diǎn)就體現(xiàn)出來(lái)了。批量注入還有一些其它的玩法,比如篩選類名,篩選父類等。
屬性注入
.Net Core 自帶的 DI 框架與 Autofac 默認(rèn)都是構(gòu)造函數(shù)注入,官方建議也是構(gòu)造函數(shù)注入。
但是有些同學(xué)可能就不喜歡構(gòu)造函數(shù)注入,再加上有些場(chǎng)景確實(shí)不適合構(gòu)造函數(shù)注入(比如基類實(shí)體),所以 Autofac 也支持屬性注入,下面我們來(lái)看看使用方法,在之前批量注入的基礎(chǔ)上,我們簡(jiǎn)單改造一下:
Assembly service = Assembly.Load("Autofac.Service"); Assembly iservice = Assembly.Load("Autofac.Service"); builder.RegisterAssemblyTypes(service, iservice) .Where(t => t.FullName.EndsWith("Service") && !t.IsAbstract) .InstancePerLifetimeScope() .AsImplementedInterfaces() .PropertiesAutowired(); // 屬性注入
對(duì)比構(gòu)造函數(shù)注入,屬性注入就多追加了PropertiesAutowired() 函數(shù),控制器內(nèi)修改:
public IUserService userService { get; set; } public IProductService productService { get; set; }
注意:屬性注入記得將屬性的訪問(wèn)修飾符改為注冊(cè)類可訪問(wèn)的修飾符,否則會(huì)注入失敗。
下面我們來(lái)看看使用效果:
咦,怎么會(huì)空引用呢?原因大概就是 Controller 是由Mvc 模塊管理的,不在 IOC 容器內(nèi),所以在 Controller 中無(wú)法使用 Autofac 注入的實(shí)例。
那為什么構(gòu)造函數(shù)注入的時(shí)候又可以呢?大概或許可能他們都是構(gòu)造函數(shù)注入吧...
為什么是大概呢?因?yàn)槲視簳r(shí)也沒(méi)有具體去深入研究到底是什么原因?qū)е碌模绻幸惶?,我想起?lái)去研究了并且有結(jié)果了,我會(huì)在這里補(bǔ)上。
我們先解決上面的問(wèn)題先,在 Startup 的 ConfigureServices 方法底部加入如下代碼:
// 使用 ServiceBasedControllerActivator 替換 DefaultControllerActivator; // Controller 默認(rèn)是由 Mvc 模塊管理的,不在 Ioc 容器中。替換之后,將放在 Ioc 容器中。 services.Replace(ServiceDescriptor.Transient<IControllerActivator, ServiceBasedControllerActivator>());
然后回到我們的AutofacModuleRegister 注入 Controller:
builder.RegisterTypes(GetAssemblyTypes<Startup>(type => typeof(ControllerBase).IsAssignableFrom(type))) .PropertiesAutowired();
這樣處理完后,屬性注入就Ok了。
存儲(chǔ)并提取容器實(shí)例
我們?cè)谥绊?xiàng)目的基礎(chǔ)上添加兩個(gè)項(xiàng)目 Common 與 Entities,存放公共類與實(shí)體類。
我們需要在實(shí)體類里面使用到 Common 項(xiàng)目中的某個(gè)類,結(jié)構(gòu)如下:
// 基類實(shí)體 public class BaseEntity { public Class1 common_Class1 { get; set; } public string CreateId { get; set; } public void Create() { this.CreateId = common_Class1.getCurrentUserId(); } } // 公共類 public class Class1 { public string getCurrentUserId() { return Guid.NewGuid().ToString(); } }
從上面的接口中我們可以看到,我需要將 Class1 通過(guò)屬性注入到容器中:
builder.RegisterType<Class1>().PropertiesAutowired().InstancePerLifetimeScope();
我們先在 Controller 中看看效果:
public Class1 class1 { get; set; } [HttpGet] public string Autofac() { return class1.getCurrentUserId(); }
很顯然結(jié)果是沒(méi)問(wèn)題的:
那我們?cè)俚?BaseEntity 中去試試看:
咦,又出現(xiàn)空引用,注入失敗了。其實(shí)這個(gè)問(wèn)題很明顯,我們使用的是 new 來(lái)實(shí)例化的 BaseEntity對(duì)象,沒(méi)有遵循容器實(shí)例使用規(guī)則,自然就無(wú)法使用容器中的實(shí)例了。
大家可以自己試一下,將 new 改為屬性注入就沒(méi)問(wèn)題了,但是這種方案并不友好,下面要說(shuō)的是另一種方案。
我們?cè)傩绿砑右粋€(gè)公共類:ContainerHelper,并聲明一個(gè)屬性用來(lái)存儲(chǔ)容器的實(shí)例:
public static class ContainerHelper { public static ILifetimeScope ContainerBuilder { get; set; } }
然后回到 Startup 中,在Configure 方法的底部加入如下代碼:
ContainerHelper.ContainerBuilder = app.ApplicationServices.CreateScope().ServiceProvider.GetAutofacRoot();
再回到實(shí)體類中去使用:
public void Create() { if (common_Class1 == null) { using (var scope = ContainerHelper.ContainerBuilder.BeginLifetimeScope()) { common_Class1 = scope.Resolve<Class1>(); } } this.CreateId = common_Class1.getCurrentUserId(); }
Autofac 的替換方案暫時(shí)就寫到這里了,后續(xù)如果有新的理解或心得會(huì)再做修改,淺談嘛就真的是淺談,有錯(cuò)誤或補(bǔ)充的地方請(qǐng)大家不吝賜教。
源碼這里就不提供了,大家有耐心的可以跟著手敲一遍,雖然對(duì)理解沒(méi)啥作用,但能使記憶更深刻一點(diǎn)。
到此這篇關(guān)于.Net Core中使用Autofac替換自帶的DI容器的示例的文章就介紹到這了,更多相關(guān).Net Core Autofac替換DI容器內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
ASP.NET使用gridview獲取當(dāng)前行的索引值
這篇文章主要介紹了ASP.NET使用gridview獲取當(dāng)前行的索引值的方法匯總,有需要的小伙伴可以參考下。2015-06-06ASP.NET DropDownListCheckBox使用示例(解決回發(fā)問(wèn)題)
本文為大家介紹下ASP.NET DropDownListCheckBox的使用,這個(gè)是根據(jù)LigerUI改的,解決了回發(fā)問(wèn)題,喜歡的朋友可以參考下2013-11-11淺談ASP.NET Core中間件實(shí)現(xiàn)分布式 Session
這篇文章主要介紹了淺談ASP.NET Core中間件實(shí)現(xiàn)分布式 Session,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2017-11-11C#反射實(shí)例學(xué)習(xí)及注意內(nèi)容
C#反射的入門學(xué)習(xí)首先要明白C#反射提供了封裝程序集、模塊和類型的對(duì)象等等需要的朋友可以參考下2012-12-12關(guān)于.NET動(dòng)態(tài)代理的介紹和應(yīng)用簡(jiǎn)介
關(guān)于.NET動(dòng)態(tài)代理的介紹和應(yīng)用簡(jiǎn)介...2006-09-09詳解ASP.NET MVC3:Razor的@:和語(yǔ)法
這篇文章主要介紹了詳解ASP.NET MVC3:Razor的@:和語(yǔ)法,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2019-01-01asp.net基礎(chǔ)學(xué)習(xí)之前端頁(yè)面布局
這篇文章主要為大家詳細(xì)介紹了asp.net基礎(chǔ)學(xué)習(xí)之前端頁(yè)面布局,什么是母版頁(yè),如何創(chuàng)建母版頁(yè),感興趣的小伙伴們可以參考一下2016-08-08asp.net post方法中參數(shù)取不出來(lái)的解決方法
調(diào)試client端調(diào)用web api的代碼,服務(wù)器端的post方法的參數(shù)死活取不出來(lái),下面有個(gè)不錯(cuò)的解決方法,希望對(duì)大家有所幫助2014-01-01