.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)(data:image/gif;base64,R0lGODlhAQABAPABAP///wAAACH5BAEKA ...
  • 今日所學: /* 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
  • 示例項目結構 在 Visual Studio 中創建一個 WinForms 應用程式後,項目結構如下所示: MyWinFormsApp/ │ ├───Properties/ │ └───Settings.settings │ ├───bin/ │ ├───Debug/ │ └───Release/ ...
  • [STAThread] 特性用於需要與 COM 組件交互的應用程式,尤其是依賴單線程模型(如 Windows Forms 應用程式)的組件。在 STA 模式下,線程擁有自己的消息迴圈,這對於處理用戶界面和某些 COM 組件是必要的。 [STAThread] static void Main(stri ...
  • 在WinForm中使用全局異常捕獲處理 在WinForm應用程式中,全局異常捕獲是確保程式穩定性的關鍵。通過在Program類的Main方法中設置全局異常處理,可以有效地捕獲並處理未預見的異常,從而避免程式崩潰。 註冊全局異常事件 [STAThread] static void Main() { / ...
  • 前言 給大家推薦一款開源的 Winform 控制項庫,可以幫助我們開發更加美觀、漂亮的 WinForm 界面。 項目介紹 SunnyUI.NET 是一個基於 .NET Framework 4.0+、.NET 6、.NET 7 和 .NET 8 的 WinForm 開源控制項庫,同時也提供了工具類庫、擴展 ...
  • 說明 該文章是屬於OverallAuth2.0系列文章,每周更新一篇該系列文章(從0到1完成系統開發)。 該系統文章,我會儘量說的非常詳細,做到不管新手、老手都能看懂。 說明:OverallAuth2.0 是一個簡單、易懂、功能強大的許可權+可視化流程管理系統。 有興趣的朋友,請關註我吧(*^▽^*) ...
  • 一、下載安裝 1.下載git 必須先下載並安裝git,再TortoiseGit下載安裝 git安裝參考教程:https://blog.csdn.net/mukes/article/details/115693833 2.TortoiseGit下載與安裝 TortoiseGit,Git客戶端,32/6 ...
  • 前言 在項目開發過程中,理解數據結構和演算法如同掌握蓋房子的秘訣。演算法不僅能幫助我們編寫高效、優質的代碼,還能解決項目中遇到的各種難題。 給大家推薦一個支持C#的開源免費、新手友好的數據結構與演算法入門教程:Hello演算法。 項目介紹 《Hello Algo》是一本開源免費、新手友好的數據結構與演算法入門 ...
  • 1.生成單個Proto.bat內容 @rem Copyright 2016, Google Inc. @rem All rights reserved. @rem @rem Redistribution and use in source and binary forms, with or with ...
  • 一:背景 1. 講故事 前段時間有位朋友找到我,說他的窗體程式在客戶這邊出現了卡死,讓我幫忙看下怎麼回事?dump也生成了,既然有dump了那就上 windbg 分析吧。 二:WinDbg 分析 1. 為什麼會卡死 窗體程式的卡死,入口門檻很低,後續往下分析就不一定了,不管怎麼說先用 !clrsta ...
  • 前言 人工智慧時代,人臉識別技術已成為安全驗證、身份識別和用戶交互的關鍵工具。 給大家推薦一款.NET 開源提供了強大的人臉識別 API,工具不僅易於集成,還具備高效處理能力。 本文將介紹一款如何利用這些API,為我們的項目添加智能識別的亮點。 項目介紹 GitHub 上擁有 1.2k 星標的 C# ...