C# IoC學習筆記

来源:https://www.cnblogs.com/atomy/archive/2020/03/20/12516304.html
-Advertisement-
Play Games

一、引言 IoC-Invertion of Control,即控制反轉,是一種程式設計思想。 先初步瞭解幾個概念: 依賴(Dependency):就是有聯繫,表示一個類依賴於另一個類。 依賴倒置原則(DIP):設計模式六大原則之一,是一種軟體架構設計原則。 控制反轉(IoC):一種軟體設計原則,上層 ...


    一、引言

    IoC-Invertion of Control,即控制反轉,是一種程式設計思想。

    先初步瞭解幾個概念:

    依賴(Dependency):就是有聯繫,表示一個類依賴於另一個類。

    依賴倒置原則(DIP):設計模式六大原則之一,是一種軟體架構設計原則。

    控制反轉(IoC):一種軟體設計原則,上層對下層的依賴(即底層模塊的獲得)交給第三方。

    依賴註入(DI):實現IoC的一種方式、手段。

    IoC容器:依賴註入的框架,用來映射依賴,管理對象的創建和生存周期。

    二、依賴

    依賴就是有聯繫,有地方使用它就是有依賴它,下麵看一個簡單的示例:

    class Program
    {
        class BMW
        {
            public string Show()
            {
                return "寶馬";
            }
        }
        class ChinesePeople
        {
            private BMW bmw = new BMW();
            public void Run()
            {
                Console.WriteLine($"今天開{bmw.Show()}上班");
            }
        }

        static void Main(string[] args)
        {
            ChinesePeople people = new ChinesePeople();
            BMW bmw = new BMW();
            people.Run();
            Console.Read();
        }
    }
View Code

    上面中國人開著寶馬去上班,客戶端有使用中國人、寶馬汽車兩個對象,中國人中有使用對象寶馬汽車,我們可以從中找到三個依賴關係:

    客戶端依賴對象ChinesePeople;

    客戶端依賴對象BMW;

    ChinesePeople依賴對象BMW;

   三、依賴倒置原則

    過些日子來了新需求,中國人不僅要開寶馬去上班,還要開賓士去上班,如果按照上面直接依賴關係的方式去做,我們就需要修改ChinesePeople類,讓它實現一個參數為寶馬的重載方法Run()。顯然這樣不是好的設計,我們總不能每次新增一種汽車(即修改下層模塊)都要去修改ChinesePeople類吧(相對於汽車為上層模塊),太麻煩了。。。

    先簡單分析一下,耦合關係就是依賴關係,如果依賴關係很重,牽一發而動全身,將很難維護擴展,耦合關係越少,系統會越穩定,因此要較少依賴。

    定義:

        A.高層模塊不應依賴於底層模塊,兩者應該依賴於抽象。

        B.抽象不應該依賴於細節,細節應該依賴於抽象。

    在這個圖中,我們發現高層模塊定義介面,將不直接依賴於下層模塊,下層模塊負責實現高層模塊定義的介面,下麵看一下示例:

    class Program
    {
        interface ICar
        {
            string Show();
        }

        class BMW : ICar
        {
            public string Show()
            {
                return "寶馬";
            }
        }

        class BenZ : ICar
        {
            public string Show()
            {
                return "賓士";
            }
        }

        interface IPeople
        {
            void Run(ICar car);
        }

        class ChinesePeople : IPeople
        {
            public void Run(ICar car)
            {
                Console.WriteLine($"今天開{car.Show()}上班");
            }
        }

        static void Main(string[] args)
        {
            ICar bmw = new BMW();
            ICar benz = new BenZ();
            IPeople people = new ChinesePeople();
            people.Run(bmw);
            people.Run(benz);
            Console.Read();
        }
    }
View Code

    運行結果如下:

    分析:上面代碼中,ChinesePeople類不再依賴於具體的汽車,而是依賴於汽車的抽象,這樣使得不管換什麼樣的汽車品牌,中國人都是可以開著去上班的,而且不需要修改ChinesePeople類。想一下,這樣是不是挺好的,我們可以得出:上層不再依賴細節,相比面向實現,面向介面較好,因為抽象相比細節要更穩定。

    四、控制反轉

    上面示例中,我們實現了具體的人和具體的汽車的隔離,具體人只和汽車的介面有關。但是Program中Main方法里的具體對象寫死了,控制權變小,當我要修改美國人開著福特去上班時,就不得不要去修改代碼,那怎麼把控制權轉移呢?

    下麵看一個簡單的示例(請先添加System.Configuration引用):

    interface ICar
    {
        string Show();
    }
ICar.cs
    interface IPeople
    {
        void Run(ICar car);
    }
IPeople.cs
    class BMW : ICar
    {
        public string Show()
        {
            return "寶馬";
        }
    }
BMW.cs
    class ChinesePeople : IPeople
    {
        public void Run(ICar car)
        {
            Console.WriteLine($"今天開{car.Show()}上班");
        }
    }
ChinesePeople.cs
<?xml version="1.0" encoding="utf-8" ?>
<configuration>
    <startup> 
        <supportedRuntime version="v4.0" sku=".NETFramework,Version=v4.6.1" />
    </startup>
    <appSettings>
        <add key="People" value="LinkTo.Test.ConsoleIoC.ChinesePeople,LinkTo.Test.ConsoleIoC"/>
        <add key="Car" value="LinkTo.Test.ConsoleIoC.BMW,LinkTo.Test.ConsoleIoC"/>
    </appSettings>
</configuration>
App.config
    class Program
    {
        static void Main(string[] args)
        {
            #region 反射+配置文件實現Ioc
            string people = ConfigurationManager.AppSettings["People"];
            string car = ConfigurationManager.AppSettings["Car"];
            Assembly assemblyPeople = Assembly.Load(people.Split(',')[1]);
            Assembly assemblyCar = Assembly.Load(car.Split(',')[1]);
            Type typePeople = assemblyPeople.GetType(people.Split(',')[0]);
            Type typeCar = assemblyPeople.GetType(car.Split(',')[0]);
            IPeople ipeople = (IPeople)Activator.CreateInstance(typePeople);
            ICar icar = (ICar)Activator.CreateInstance(typeCar);
            ipeople.Run(icar);
            Console.Read();
            #endregion
        }
    }
Program.cs

    上面代碼中,我們使用反射+配置文件的方式,將對象創建的控制權轉移到了配置文件,這就是所謂的控制反轉

    分析:控制反轉是將對象創建的控制權交給了第三方,可以是IoC容器,它就相當於簡單工廠。我們要什麼對象,工廠就給我們什麼對象,這樣依賴關係就變了,它們(人和車)都依賴於IoC容器,通過IoC容器建立它們之間的依賴關係。(依賴對象不再直接通過new來獲取)

    運行結果如下:

    五、依賴註入

    上面說到的控制反轉,我們瞭解到是將控制權轉移,這是我們的目的。配置文件+反射是一種實現,而依賴註入則提供的是一種思想,或者說是實現IoC的手段。

    依賴註入是將對象的創建和綁定轉移到被依賴對象的外部來實現,一般使用哪些方法來實現呢?

    方法一:構造函數註入

    class ChinesePeopleConstructor
    {
        private readonly ICar _car;

        //依賴註入:構造函數註入
        public ChinesePeopleConstructor(ICar car)
        {
            _car = car;
        }

        public void Run()
        {
            Console.WriteLine($"今天開{_car.Show()}上班");
        }
    }
ChinesePeopleConstructor.cs
    class Program
    {
        static void Main(string[] args)
        {
            #region 依賴註入:構造函數註入
            ICar bmw = new BMW();
            ChinesePeopleConstructor people = new ChinesePeopleConstructor(bmw);
            people.Run();
            Console.Read();
            #endregion
        }
    }
Program.cs

    方法二:屬性註入

    class ChinesePeopleProperty
    {
        //依賴註入:屬性註入
        public ICar Car { get; set; }

        public void Run()
        {
            Console.WriteLine($"今天開{Car.Show()}上班");
        }
    }
ChinesePeopleProperty.cs
    class Program
    {
        static void Main(string[] args)
        {
            #region 依賴註入:屬性註入
            ICar bmw = new BMW();
            ChinesePeopleProperty people = new ChinesePeopleProperty
            {
                Car = bmw
            };
            people.Run();
            Console.Read();
            #endregion
        }
    }
Program.cs

    方法三:介面註入

    interface IDependent
    {
        void SetDependent(ICar icar);
    }
IDependent.cs
    class ChinesePeopleInterface : IDependent
    {
        private ICar _car;

        //依賴註入:介面註入
        public void SetDependent(ICar car)
        {
            _car = car;
        }

        public void Run()
        {
            Console.WriteLine($"今天開{_car.Show()}上班");
        }
    }
ChinesePeopleInterface.cs
    class Program
    {
        static void Main(string[] args)
        {
            #region 依賴註入:介面註入
            ICar bmw = new BMW();
            ChinesePeopleInterface people = new ChinesePeopleInterface();
            people.SetDependent(bmw);
            people.Run();
            Console.Read();
            #endregion
        }
    }
Program.cs

    六、IoC容器

    IoC容器是一個DI框架,主要功能有一下幾點:

    A.動態創建、註入依賴對象;

    B.管理對象生命周期;

    C.映射依賴關係;

    常見的IoC容器:Spring.NET,Castle Windsor, Ninject,Autofac,Unity。。。

    6.1、Unity容器使用

    在上一篇《C# AOP學習筆記》的【使用EntLib\PIAB Unity實現AOP(帶配置)】中,已經使用Unity容器實現了IoC,讓我們再來看看配置文件:

    假如只需要IoC不需要AOP,container是這樣子的:

<configuration>
  <configSections>
    <section name="unity" type="Microsoft.Practices.Unity.Configuration.UnityConfigurationSection, Unity.Configuration"/>
  </configSections>
  <unity>
    <sectionExtension type="Microsoft.Practices.Unity.InterceptionExtension.Configuration.InterceptionConfigurationExtension, Unity.Interception.Configuration"/>
    <containers>
      <container name="IoCContainer">
        <!--註冊匹配規則:前面是完整類型名稱,後面是所在的dll名稱。-->
        <register type="LinkTo.Test.ConsoleAop.UnityConfigAOP.IUserProcessor,LinkTo.Test.ConsoleAop" mapTo="LinkTo.Test.ConsoleAop.UnityConfigAOP.UserProcessor,LinkTo.Test.ConsoleAop"></register>
      </container>
    </containers>
  </unity>
</configuration>
View Code

    從IoC的註冊匹配規則可以看出,前面是完整類型名稱,後面是所在的dll。這個就比較厲害了,假如一個系統有擴展的功能或者個性要求,只需配置使用新的dll即可,原有系統不需要改代碼。這樣,除了很好地符合了"開閉原則"外,對於構建一個可配置可擴展的系統,是一個非常厲害的利器。 

 

    參考自:

    https://www.cnblogs.com/jdzhang/p/7104351.html

    推薦博文:

    基於介面設計三層架構


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

-Advertisement-
Play Games
更多相關文章
  • 字元串 string是數據類型,不是引用或者指針類型 string是只讀的byte slice,len函數可以獲取他所有的byte數量 string的byte數組可以存放任何數據 輸出 註意:len獲取的string的byte個數,不是字元數 Unicode UTF8 Unicode是一種字元集(c ...
  • Python中文命名 命名規則數字、字母、下劃線 環境 1. python 3.7.3 x64 2. win10 現象 Python在命名的地方都是可以用中文命名 如: 1. 變數 1. 中文下劃線無法被識別 2. 打出中文下劃線按退格鍵會不會被識別,會把前邊一個字元刪了 2. 類、類屬性、方法、函 ...
  • 本人免費整理了Java高級資料,涵蓋了Java、Redis、MongoDB、MySQL、Zookeeper、Spring Cloud、Dubbo高併發分散式等教程,一共30G,需要自己領取。傳送門:https://mp.weixin.qq.com/s/osB-BOl6W-ZLTSttTkqMPQ 誰 ...
  • 導讀 前幾天和一個朋友討論了他們公司的系統問題,傳統的單體應用,集群部署,他說近期服務的併發量可能會出現瞬時增加的風險,雖然部署了集群,但是通過壓測後發現請求延遲仍然是很大,想問問我有什麼改進的地方。我沉思了一會,現在去改架構顯然是不可能的,於是我給出了一個建議,讓他去做個介面限流,這樣能夠保證瞬時 ...
  • 前言 在前一篇文章中,我提到最近要陸續為大家寫一些.Net實戰技術文章。從今天起,我將圍繞一個入門級現實的芒果分銷管理系統案例,使用ASP.NET MVC 5,從前端到後端,一步一步為大家呈現整個系統的開發過程與業務關鍵架構以及代碼。 如果您是一位.Net初學者 如果您剛剛接觸MVC 如果您剛剛接觸 ...
  • ArcGISRuntime 載入 高德、騰訊、百度地圖 瓦片並顯示。 環境# Visual Studio 2019,dotNet Framework 4.6.1 SDK 支持Windows Win7、8、10 源碼地址 效果# ...
  • 具體要求為: 使用一個二維數組記錄客車售票系統中的所有座位號,併在每個座位號上都顯示有票,然後用戶輸入一個坐標位置,按Enter鍵,即可將該座位號顯示為已售。 首先我定義的輸入格式為:1,2 個人認為主要知識點偽代碼如下 1.字元串分割 char[] separator = { ',' }; spl ...
  • 為了開發規範,有時需要統一響應屬性名稱,.netcore已為我們封裝好了,我們直接用即可。 在StartUp類中ConfigureServices方法中,添加如下代碼: public void ConfigureServices(IServiceCollection services) { serv ...
一周排行
    -Advertisement-
    Play Games
  • 前言 在我們開發過程中基本上不可或缺的用到一些敏感機密數據,比如SQL伺服器的連接串或者是OAuth2的Secret等,這些敏感數據在代碼中是不太安全的,我們不應該在源代碼中存儲密碼和其他的敏感數據,一種推薦的方式是通過Asp.Net Core的機密管理器。 機密管理器 在 ASP.NET Core ...
  • 新改進提供的Taurus Rpc 功能,可以簡化微服務間的調用,同時可以不用再手動輸出模塊名稱,或調用路徑,包括負載均衡,這一切,由框架實現並提供了。新的Taurus Rpc 功能,將使得服務間的調用,更加輕鬆、簡約、高效。 ...
  • 順序棧的介面程式 目錄順序棧的介面程式頭文件創建順序棧入棧出棧利用棧將10進位轉16進位數驗證 頭文件 #include <stdio.h> #include <stdbool.h> #include <stdlib.h> 創建順序棧 // 指的是順序棧中的元素的數據類型,用戶可以根據需要進行修改 ...
  • 前言 整理這個官方翻譯的系列,原因是網上大部分的 tomcat 版本比較舊,此版本為 v11 最新的版本。 開源項目 從零手寫實現 tomcat minicat 別稱【嗅虎】心有猛虎,輕嗅薔薇。 系列文章 web server apache tomcat11-01-官方文檔入門介紹 web serv ...
  • C總結與剖析:關鍵字篇 -- <<C語言深度解剖>> 目錄C總結與剖析:關鍵字篇 -- <<C語言深度解剖>>程式的本質:二進位文件變數1.變數:記憶體上的某個位置開闢的空間2.變數的初始化3.為什麼要有變數4.局部變數與全局變數5.變數的大小由類型決定6.任何一個變數,記憶體賦值都是從低地址開始往高地 ...
  • 如果讓你來做一個有狀態流式應用的故障恢復,你會如何來做呢? 單機和多機會遇到什麼不同的問題? Flink Checkpoint 是做什麼用的?原理是什麼? ...
  • C++ 多級繼承 多級繼承是一種面向對象編程(OOP)特性,允許一個類從多個基類繼承屬性和方法。它使代碼更易於組織和維護,並促進代碼重用。 多級繼承的語法 在 C++ 中,使用 : 符號來指定繼承關係。多級繼承的語法如下: class DerivedClass : public BaseClass1 ...
  • 前言 什麼是SpringCloud? Spring Cloud 是一系列框架的有序集合,它利用 Spring Boot 的開發便利性簡化了分散式系統的開發,比如服務註冊、服務發現、網關、路由、鏈路追蹤等。Spring Cloud 並不是重覆造輪子,而是將市面上開發得比較好的模塊集成進去,進行封裝,從 ...
  • class_template 類模板和函數模板的定義和使用類似,我們已經進行了介紹。有時,有兩個或多個類,其功能是相同的,僅僅是數據類型不同。類模板用於實現類所需數據的類型參數化 template<class NameType, class AgeType> class Person { publi ...
  • 目錄system v IPC簡介共用記憶體需要用到的函數介面shmget函數--獲取對象IDshmat函數--獲得映射空間shmctl函數--釋放資源共用記憶體實現思路註意 system v IPC簡介 消息隊列、共用記憶體和信號量統稱為system v IPC(進程間通信機制),V是羅馬數字5,是UNI ...