ASP.NET Core依賴關(guān)系注入
1.前言
面向?qū)ο笤O(shè)計(jì)(OOD)里有一個(gè)重要的思想就是依賴倒置原則(DIP),并由該原則牽引出依賴注入(DI)、控制反轉(zhuǎn)(IOC)及其容器等老生常談的概念,初學(xué)者很容易被這些概念搞暈(包括我在內(nèi)),在學(xué)習(xí)Core依賴注入服務(wù)之前,下面讓我們先了解下依賴倒置原則(DIP)、依賴注入(DI)、控制反轉(zhuǎn)(IOC)等概念,然后再深入學(xué)習(xí)Core依賴注入服務(wù)。
2.依賴倒置原則(DIP)
高層模塊不依賴于低層模塊的實(shí)現(xiàn),而低層模塊依賴于高層模塊定義的接口。通俗來(lái)講,就是高層模塊定義接口,低層模塊負(fù)責(zé)實(shí)現(xiàn)。
2.依賴注入(DI)
2.1依賴(D)
當(dāng)一個(gè)類需要另一個(gè)類協(xié)作來(lái)完成工作的時(shí)候就產(chǎn)生了依賴。
示例1:
public class MyDependency { public MyDependency() { } public Task WriteMessage(string message) { Console.WriteLine($"MyDependency.WriteMessage called. Message: {message}"); return Task.FromResult(0); } } public class IndexModel : PageModel { MyDependency _dependency = new MyDependency(); public void OnGet() { _dependency.WriteMessage("IndexModel.OnGet created this message."); } }
由上述代碼可以看到IndexModel模塊輸出消息必須要實(shí)例化MyDependency模塊,也就是說(shuō)IndexModel模塊業(yè)務(wù)的實(shí)現(xiàn)必須依賴于MyDependency模塊,這就是依賴。
2.2 注入(I)
根據(jù)DIP設(shè)計(jì)原則:高層模塊不依賴于低層模塊的實(shí)現(xiàn),而低層模塊依賴于高層模塊定義的接口,所以我們?cè)谶@里定義一個(gè)接口供高層模塊調(diào)用,底層模塊負(fù)責(zé)實(shí)現(xiàn)。
示例2:
public interface IMyDependency { Task WriteMessage(string message); } public class MyDependency: IMyDependency { public MyDependency() { } public Task WriteMessage(string message) { Console.WriteLine($"MyDependency.WriteMessage called. Message: {message}"); return Task.FromResult(0); } } public class IndexModel : PageModel { IMyDependency _dependency = new MyDependency(); public void OnGet() { _dependency.WriteMessage("IndexModel.OnGet created this message."); } }
從上述代碼可以看到當(dāng)我們調(diào)用IndexModel模塊OnGetAsync方法的時(shí)候,是通過(guò)IMyDependency接口實(shí)例化MyDependency類去實(shí)現(xiàn)其方法內(nèi)容的,這叫控制正轉(zhuǎn)。但是Master說(shuō),我們不應(yīng)該創(chuàng)建MyDependency類,而是讓調(diào)用者給你傳遞,于是你通過(guò)構(gòu)造函數(shù)讓外界把這兩個(gè)依賴給你。
示例3:
public interface IMyDependency { Task WriteMessage(string message); } public class MyDependency : IMyDependency { private readonly ILogger<MyDependency> _logger; public MyDependency(ILogger<MyDependency> logger) { _logger = logger; } public Task WriteMessage(string message) { _logger.LogInformation( "MyDependency.WriteMessage called. Message: {MESSAGE}", message); return Task.FromResult(0); } } public class IndexModel : PageModel { private readonly IMyDependency _dependency; public IndexModel(IMyDependency dependency) { _dependency = dependency; } public void OnGet() { _dependency.WriteMessage("IndexModel.OnGet created this message."); } }
從上述代碼可以看到把依賴的創(chuàng)建丟給第三方系統(tǒng)(例:Autofac,Unity容器),也叫控制反轉(zhuǎn)(IOC)容器。自己只負(fù)責(zé)使用,其它人丟給你依賴的這個(gè)過(guò)程理解為注入。也叫控制反轉(zhuǎn)(IOC)。注意,框架內(nèi)部ILogger接口已注入,無(wú)需手動(dòng)再重新注入。
2.3 IOC容器
IOC容器可以看作是負(fù)責(zé)統(tǒng)一管理依賴關(guān)系的地方。常見有Autofac,Unity。
容器只要負(fù)責(zé)兩件事情:
●綁定服務(wù)與實(shí)例之間的關(guān)系
●獲取實(shí)例,并對(duì)實(shí)例進(jìn)行管理(創(chuàng)建與銷毀)
3.依賴倒置原則(DIP)與控制反轉(zhuǎn)(IOC)的區(qū)別
DIP是一種軟件設(shè)計(jì)原則,它僅僅告訴你高低層模塊之間應(yīng)該如何依賴,但是它并沒有告訴我們?nèi)绾谓獬嗷ヒ蕾嚹K的耦合。而IOC則是一種軟件設(shè)計(jì)模式,它告訴我們?cè)撊绾谓獬K的耦合,它為相互依賴的組件提供抽象,將依賴(低層模塊)對(duì)象的獲得交給第三方系統(tǒng)(例:Autofac,Unity容器)來(lái)控制,即依賴對(duì)象不在被依賴模塊的類中直接通過(guò)new來(lái)獲取。
4.NET Core依賴注入(DI)服務(wù)
經(jīng)過(guò)上面描述,大家應(yīng)該應(yīng)該對(duì)依賴倒置原則(DIP)、依賴注入(DI)、控制反轉(zhuǎn)(IOC)這幾個(gè)概念有一定了解對(duì)吧。在.NET Core中DI的核心分為兩個(gè)組件:IServiceCollection和 IServiceProvider。
- IServiceCollection負(fù)責(zé)注冊(cè)
- IServiceProvider負(fù)責(zé)提供實(shí)例
下面讓我們來(lái)學(xué)習(xí)下NET Core是怎么依賴注入(DI)服務(wù)。
第一步:使用接口來(lái)實(shí)現(xiàn)依賴反轉(zhuǎn)。定義 IMyDependency 服務(wù)。
public interface IMyDependency { Task WriteMessage(string message); }
第二步:定義IMyDependency 服務(wù)的實(shí)現(xiàn)類MyDependency。
public class MyDependency : IMyDependency { private readonly ILogger<MyDependency> _logger; public MyDependency(ILogger<MyDependency> logger) { _logger = logger; } public Task WriteMessage(string message) { _logger.LogInformation( "MyDependency.WriteMessage called. Message: {MESSAGE}", message); return Task.FromResult(0); } }
第三步:把IMyDependency 服務(wù)注冊(cè)到服務(wù)容器中。
public void ConfigureServices(IServiceCollection services) { //注冊(cè)將服務(wù)生命期的范圍限定為單個(gè)請(qǐng)求的生命期,下節(jié)再來(lái)聊服務(wù)生命期 services.AddScoped<IMyDependency, MyDependency>(); }
第四步:把服務(wù)注入到使用它的類的構(gòu)造函數(shù)中。在HomeController里面調(diào)用IndexModel.OnGet方法輸出WriteMessage消息。
public class IndexModel : PageModel { private readonly IMyDependency _dependency; public IndexModel(IMyDependency dependency) { _dependency = dependency; } public void OnGet() { _dependency.WriteMessage("IndexModel.OnGet created this message."); } } private readonly IMyDependency _iMyDependency; public HomeController(IMyDependency iMyDependency) { _iMyDependency = iMyDependency; } public IActionResult Index() { IndexModel _IndexModel = new IndexModel(_iMyDependency); _IndexModel.OnGet(); return View(); }
WriteMessage日志消息如下:
5.默認(rèn)服務(wù)容器替換
下面我們將來(lái)演示內(nèi)置容器怎么替換為其他容器示例,比如替換第三方 Autofac容器,我們需要在Startup.ConfigureServices方法里面注冊(cè)Autofac容器,具體代碼如下:
public IServiceProvider ConfigureServices(IServiceCollection services) { // Add Autofac var containerBuilder = new ContainerBuilder(); containerBuilder.RegisterModule<DefaultModule>(); containerBuilder.Populate(services); var container = containerBuilder.Build(); return new AutofacServiceProvider(container); }
這里需要注意的是如果需要使用第三方容器,Startup.ConfigureServices 必須返回 IServiceProvider。然后自定義一個(gè)模塊類配置依賴關(guān)系,具體代碼如下:
public class DefaultModule : Module { protected override void Load(ContainerBuilder builder) { builder.RegisterType<CharacterRepository>().As<ICharacterRepository>(); } }
應(yīng)用程序在運(yùn)行時(shí),使用 Autofac 來(lái)解析類型,并注入依賴關(guān)系。
到此這篇關(guān)于ASP.NET Core依賴關(guān)系注入的文章就介紹到這了。希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
ASP.NET Core MVC中的控制器(Controller)介紹
這篇文章介紹了ASP.NET Core MVC中的控制器(Controller),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2022-04-04一步步打造漂亮的新聞列表(無(wú)刷新分頁(yè)、內(nèi)容預(yù)覽)第一步
新聞列表是信息管理系統(tǒng)中最常見的,也是最簡(jiǎn)單的,一些簡(jiǎn)單的新聞列表就是一個(gè)table,然后里面循環(huán)寫入數(shù)據(jù)2010-07-07ASP.NET中DropDownList下拉框列表控件綁定數(shù)據(jù)的4種方法
本文主要介紹了DropDownList控件4種綁定數(shù)據(jù)的基礎(chǔ)用法,希望對(duì)大家能有所幫助。2016-04-04