.NET Core ASP.NET Core Basic 1-2 控制反轉與依賴註入

来源:https://www.cnblogs.com/WarrenRyan/archive/2019/09/02/11444398.html
-Advertisement-
Play Games

.NET Core ASP.NET Core Basic 1 2 本節內容為控制反轉與依賴註入 簡介 控制反轉IOC 這個內容事實上在我們的C 高級篇就已經有所講解,控制反轉是一種設計模式,你可以這樣理解控制反轉,假設有一個人他有一部A品牌手機,他用手機進行聽歌、打游戲,那麼你可以創建一個手機類和一 ...


.NET Core ASP.NET Core Basic 1-2

本節內容為控制反轉與依賴註入

簡介

控制反轉IOC

這個內容事實上在我們的C#高級篇就已經有所講解,控制反轉是一種設計模式,你可以這樣理解控制反轉,假設有一個人他有一部A品牌手機,他用手機進行聽歌、打游戲,那麼你可以創建一個手機類和一個人類

class APhone : IPhone
{
    public string Owner{get;set;}
    public Phone(string own)
    {
        Owner = own;
    }
    void Play()
    {
        //省略
    }
    void Music()
    {
        //省略
    }
}
class Man
{
    public string Name{get;set;}
    void Game()
    {
        var p = new APhone(Name);
        p.Play();
    }
}

事實上這段代碼的耦合度是比較高的?它使用的是正轉,也就是我需要什麼東西的時候我就自己創建一個這個東西。為什麼說他不好呢,如果有一天這個人決定再也不使用A品牌手機了,他決定以後只使用B品牌。那麼也就意味著整個的Man類使用過APhone類的地方都需要更改。這是一個非常麻煩的事情,我們這個時候就需要運用我們的IOC控制反轉了。我們將實例或者是需要使用的對象的創建交給你的調用者,自己只負責使用,其它人丟給你依賴的這個過程理解為註入。

控制反轉的核心就是——原本我保存使用我自己的東西,現在我把控制權交給我的上級,我需要使用的時候再向他要。這個時候,介面的作用不言而喻,A繼承了Phone介面,B也繼承了,假定我們一開始就使用Phone介面去創建不同的A,B對象,那麼是不是可以有效的切換AB對象呢?

依賴註入

依賴註入體現的是一個IOC(控制反轉),它非常的簡單,我們之前的Man類代碼中使用的是正轉的方式,也就是我要一個對象,那麼我就創建一個。現在我們使用依賴註入就是將我們對這個對象的控制權交給上一級介面,也就成為了這種,我想要一個對象,我就向上級發出請求,上級就給我創建了一個對象。我們通常使用構造函數註入的方式進行依賴的註入。

上文的代碼就會變成

class Man
{
    private readonly IPhone _phone;
    public Man(IPhone phone)
    {
        _phone = phone;
    }
}

假設這個時候你需要將手機換成B品牌,那麼只需要再註入的地方傳入B品牌的對象即可了。

容器

但是現在又出現了一個新的問題,假設說這個類有100個使用該介面的依賴,如果,我們是不是要在100個地方做這樣的事情? 控制是反轉了,依賴的創建也移交到了外部。現在的問題是依賴太多,我們需要一個地方統一管理系統中所有的依賴,這個時候,我們就使用容器進行集中的管理

容器負責兩件事情:

  • 綁定服務與實例之間的關係
  • 獲取實例,並對實例進行管理(創建與銷毀)

使用

說了那麼多,我們如何在.NET Core中使用我們的依賴註入呢?這裡我們針對的是所有的.NET Core的應用,在.NET Core中依賴註入的核心分為兩個組件:位於Microsoft.Extensions.DependencyInjection命名空間下的IServiceCollection和 IServiceProvider。

其中

  • IServiceCollection 負責註冊
  • IServiceProvider 負責提供實例

在預設的容器ServiceCollection中有三個方法

  • .AddTransient<I,C>()
  • .AddSingleton<I,C>()
  • .AddScoped<I,C>()

這裡就不得不提一下我們依賴註入的三種生命周期了

  • Singleton指的是單例模式,也就是說,在整個程式運轉期間只會生成一次
  • Transient指每一次GetService都會創建一個新的實例
  • Scope指在同一個Scope內只初始化一個實例 ,可以理解為( 每一個request級別隻創建一個實例,同一個http request會在一個 scope內)

我們可以嘗試使用控制台項目來模擬依賴註入的原理,也就是說我們直接從容器獲取我們對象實例,並且我們使用Guid進行唯一性的標記。

//創建三個代表不同生命周期的介面
    interface IPhoneScope
    {
        Guid Guid { get; }
    }
    interface IPhoneSingleton
    {
        Guid Guid { get; }
    }
    interface IPhoneTransient
    {
        Guid Guid { get; }
    }
    //實現的類
    class PhoneService:IPhoneScope,IPhoneSingleton,IPhoneTransient
    {
        public PhoneService()
        {
            this._guid = Guid.NewGuid();
        }

        public PhoneService(Guid guid)
        {
            this._guid = guid;
        }

        private Guid _guid;

        public Guid Guid => this._guid;
    }

然後,在我們的主函數中

namespace DI_AND_IOC
{
    class Program
    {
        static void Main(string[] args)
        {
            //註入服務
            var services = new ServiceCollection()
                .AddScoped<IPhoneScope, PhoneService>()
                .AddTransient<IPhoneTransient, PhoneService>()
                .AddSingleton<IPhoneSingleton, PhoneService>();
            //構造服務
            var provider = services.BuildServiceProvider();
            using (var scope = provider.CreateScope())
            {
                var p = scope.ServiceProvider;
                var scopeobj1 = p.GetService<IPhoneScope>();
                var transient1 = p.GetService<IPhoneTransient>();
                var singleton1 = p.GetService<IPhoneSingleton>();

                var scopeobj2 = p.GetService<IPhoneScope>();
                var transient2 = p.GetService<IPhoneTransient>();
                var singleton2 = p.GetService<IPhoneSingleton>();

                Console.WriteLine(
                    $"scope1: {scopeobj1.Guid},\n" +
                    $"transient1: {transient1.Guid}, \n" +
                    $"singleton1: {singleton1.Guid}\n");

                Console.WriteLine($"scope2: {scopeobj2.Guid}, \n" +
                                  $"transient2: {transient2.Guid},\n" +
                                  $"singleton2: {singleton2.Guid}\n");
            }
            //創建不同的scope
            using (var scope = provider.CreateScope())
            {
                var p = scope.ServiceProvider;
                var scopeobj3 = p.GetService<IPhoneScope>();
                var transient3 = p.GetService<IPhoneTransient>();
                var singleton3 = p.GetService<IPhoneSingleton>();
                Console.WriteLine($"scope3: {scopeobj3.Guid}, \n" +
                                  $"transient3: {transient3.Guid},\n" +
                                  $"singleton3: {singleton3.Guid}");
            }
        }
    }
}

你應該會得到類似以下的數據

scope1: 096d38e5-0c7b-4e50-9c79-241fb18a56ed,
transient1: 289ebd11-8159-4f22-b53e-ed738a317313,
singleton1: b453b7f5-3594-4b66-99c8-a72763abaa83

scope2: 096d38e5-0c7b-4e50-9c79-241fb18a56ed,
transient2: 212ad420-e54c-4dd6-9214-abe91aacdd9c,
singleton2: b453b7f5-3594-4b66-99c8-a72763abaa83

scope3: 688b6ffd-a8c1-47f4-a20a-872c2285d67c,
transient3: 3d09997d-fffb-43d1-9e53-ccf9771c819d,
singleton3: b453b7f5-3594-4b66-99c8-a72763abaa83

可以發現,singleton對象是不會發生改變的,而scope對象在創建新的scope之後就發生了改變,而transient對象每一次請求都在發生改變。

需要註意的是,在控制台項目使用容器服務需要引入 *** Microsoft.Extensions.DependencyInjection *** 程式集,你可以在引入中導入該dll

通過對註入服務的生命周期管控,在一些ASP.NET Core項目中,有些類(服務)有可能跨越了多個Action或者Controller,那麼我們正確的使用生命周期,我們可以儘可能的節省記憶體,即能減少實例初始化的消耗。

在ASP.NET Core中的使用

在ASP.NET Core中,我們使用依賴註入非常的簡單,在StartUp類中的ConfigureServices方法中已經為我們構建好了容器,我們只需要做類似於這樣的操作

services.AddScoped<IPhoneScope, PhoneService>();
services.AddDbContext<DbContext>();
services.AddMVC();

如果你需要在控制器中註入服務,官方的推薦方案是使用構造函數註入

public IPhoneScope _ips;
public Controller(IPhoneScope ips)
{
    _ips = ips;
}

特別的,你如果使用MVC的Razor頁面進行註入的話,那麼輸入以下指令

@inject IPhoneScope  ips

如果我的文章幫助了您,請您在github.NETCoreGuide項目幫我點一個star,在博客園中點一個關註和推薦。

Github

BiliBili主頁

WarrenRyan'sBlog

博客園


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

-Advertisement-
Play Games
更多相關文章
  • Flink 源碼項目結構一覽 <! more https://t.zsxq.com/MNfAYne 博客 1、 "Flink 從0到1學習 —— Apache Flink 介紹" 2、 "Flink 從0到1學習 —— Mac 上搭建 Flink 1.6.0 環境並構建運行簡單程式入門" 3、 "F ...
  • StreamGraph <! more https://t.zsxq.com/qRFIm6I 博客 1、 "Flink 從0到1學習 —— Apache Flink 介紹" 2、 "Flink 從0到1學習 —— Mac 上搭建 Flink 1.6.0 環境並構建運行簡單程式入門" 3、 "Flin ...
  • 今日所學: /* 2019.08.19開始學習,此為補檔。 */ 流程式控制制 條件: 一重用if 二重用if ... else 三重用if ... else if ... else 多重用switch 例: 迴圈: while和do ... while的區別: while是先判斷後執行,do ... ...
  • 1 下載安裝包 1.1 壓縮包 "https://dev.mysql.com/downloads/mysql/" [外鏈圖片轉存失敗(img oesO8K09 1566652568838)( ...
  • 今日所學: /* 2019.08.19開始學習,此為補檔。 */ 三目(元)運算符 格式:(表達式)? 表達式為true返回值A : 表達式為false返回值B 例: 面向對象的基本概念 定義:以基於對象的思維去分析和解決問題,萬物皆對象。 三大特性:封裝,繼承,多態。 ...
  • 以上代碼第一個if後面的內容永遠被執行。 這裡需要考慮到優先順序問題,一般而言,算數運算>關係運算>邏輯運算>位運算>賦值運算。 上面代碼中,m==1返回false,false與3進行"或"運算,3被視為true。結果返回true.同理最後返回true,所以if後的語句就一直被執行了。 常見的還有下麵 ...
  • 1、JDK https://www.oracle.com/technetwork/java/javase/downloads/jdk8 downloads 2133151.html 目前主流的JDK版本還是JAVA8,我在阿裡用的也是Java8。 JDK里已經包含了JRE也就是Java虛擬機和運行環 ...
  • 在Web系統開發過程中,文件上傳是普遍的功能,本文主要以一個簡單的小例子,講解SpringMVC中文件上傳的使用方法,僅供學習分享使用,如有不足之處,還請指正。 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...