手寫實現簡單版IOC

来源:https://www.cnblogs.com/lyps/archive/2019/03/20/10560256.html
-Advertisement-
Play Games

概述 IOC (Inversion of Control) 控制反轉,大家應該都比較熟悉了、應該也都有用過,這裡就不具體介紹了。自己平時也有用到過IOC,但是對它的具體實現原理只有一個模糊的概念,所以決定自己手動實現一個簡單IOC。 開始 首先呢我們得知道IOC的主要作用是什麼,才能開始動手寫。IO ...


概述

IOC (Inversion of Control) 控制反轉,大家應該都比較熟悉了、應該也都有用過,這裡就不具體介紹了。自己平時也有用到過IOC,但是對它的具體實現原理只有一個模糊的概念,所以決定自己手動實現一個簡單IOC。

開始

首先呢我們得知道IOC的主要作用是什麼,才能開始動手寫。IOC主要不就是負責創建對象以及管理生命周期嘛,那我們就開始動手啦。

比如現在有一個IAnimal介面Animal繼承介面,然後就是個Call的方法。一般我們使用的時候都是IAnimal animal=new Animal(); 如果是使用第三方IOC容器實現的話,我們需要先註冊一下類型才能獲取到實例。

所以我們先來個最簡單的仿照這個過程:

新建一個Container,然後裡面有一個類型註冊的方法ResgisterType和一個返回實例的方法Rerolve,還有一個存儲類型的字典,具體代碼如下

        private static Dictionary<string, object> ContainerTypeDictionary = new Dictionary<string, object>();/// <summary>
        /// 註冊類型
        /// </summary>
        /// <typeparam name="IT"></typeparam>
        /// <typeparam name="T"></typeparam>
        public void ResgisterType<IT,T>()
        {
            if (!ContainerTypeDictionary.ContainsKey(typeof(IT).FullName))
                ContainerTypeDictionary.Add(typeof(IT).FullName, typeof(T));
        }

        /// <summary>
        /// 根據註冊信息生成實例
        /// </summary>
        /// <typeparam name="IT"></typeparam>
        /// <returns></returns>
        public IT Rerolve<IT>()
        {
            string key = typeof(IT).FullName;
            Type type = (Type)ContainerTypeDictionary[key];
       return (IT)Activator.CreateInstance(type);
     }

然後我們新建一個控制台測試一下

Container container = new Container();
container.ResgisterType<IAnimal, Animal>();
IAnimal animal= container.Rerolve<IAnimal>();

然後可以在不依賴具體對象Animal的情況下成功的創建一個animal實例。

 

之後我們就可以考慮複雜一點的情況了,現在我們的Animal類里沒有做任何事,假如它的構造函數里依賴於另一個對象呢,這樣我們的程式肯定是會報錯的。比如下麵這樣:

public class Animal: IAnimal
    {
        public Animal(Dog dog)
        {
            
        }
    }

 

我們容器目前能創建的對象實例,只有通過ResgisterType方法註冊過類型的,而像Animal里依賴的不能實現創建,所以這個時候就需要用到依賴註入了。

關於依賴註入與控制反轉的關係,我個人的理解是:控制反轉是一種設計思想,而依賴註入則是實現控制反轉思想的方法。

IOC容器一般依賴註入有三種:構造函數註入、方法註入、屬性註入。

那麼我們就來照瓢畫葫蘆,實現一下構造函數註入。一般IOC容器構造函數註入是通過一個特性來識別註入的,如果沒有標記特性則去找構造函數參數個數最多的,我們就按照這個思路來。

首先我們新建一個LInjectionConstructorAttribute類,只需繼承Attribute就行了。

 

public class LInjectionConstructorAttribute :Attribute
    {
    }

 

然後在剛纔那個Animal構造函數上標記上特性,接下來就開始寫代碼。

/// <summary>
        /// 根據註冊信息生成實例
        /// </summary>
        /// <typeparam name="IT"></typeparam>
        /// <returns></returns>
        public IT Rerolve<IT>()
        {
            string key = typeof(IT).FullName;
            Type type = (Type)ContainerTypeDictionary[key];

            return (IT)CreateType(type);
        }
/// <summary>
        /// 根據提供的類型創建類型實例並返回
        /// </summary>
        /// <param name="type"></param>
        /// <returns></returns>
        private object CreateType(Type type)
        {
            var ctorArray = type.GetConstructors();
            if (ctorArray.Count(c => c.IsDefined(typeof(LInjectionConstructorAttribute), true)) > 0)
            {
                //獲取帶特性標記的構造函數參數
                foreach (var cotr in type.GetConstructors().Where(c => c.IsDefined(typeof(LInjectionConstructorAttribute), true)))
                {
                    var paraArray = cotr.GetParameters();//獲取參數數組
                    if (paraArray.Length == 0)
                    {
                        return Activator.CreateInstance(type);
                    }

                    List<object> listPara = new List<object>();
                    foreach (var para in paraArray)
                    {
                        string paraKey = para.ParameterType.FullName;//參數類型名稱
                                                                     //從字典中取出緩存的目標對象並創建對象
                        Type paraTargetType = (Type)ContainerTypeDictionary[paraKey];
                        object oPara = CreateType(paraTargetType);//遞歸
                        listPara.Add(oPara);
                    }
                    return Activator.CreateInstance(type,listPara.ToArray());
                }

                return Activator.CreateInstance(type);
            }
            else
            {
                //沒有標記特性則使用參數最多的構造函數
                var ctor = ctorArray.OrderByDescending(c => c.GetParameters().Length).FirstOrDefault();
                var paraArray = ctor.GetParameters();//獲取參數數組
                if (paraArray.Length == 0)
                {
                    return Activator.CreateInstance(type);
                }

                List<object> listPara = new List<object>();
                foreach (var para in paraArray)
                {
                    string paraKey = para.ParameterType.FullName;//參數類型名稱
                                                                 //從字典中取出緩存的目標對象並創建對象
                    Type paraTargetType = (Type)ContainerTypeDictionary[paraKey];
                    object oPara = CreateType(paraTargetType);//遞歸
                    listPara.Add(oPara);
                }
                return Activator.CreateInstance(type, listPara.ToArray());
            }
        }

這裡說下為什麼用到遞歸,在我們項目中使用會有層層依賴的關係。比如我這裡Animal依賴於Dog只有一層依賴,如果Gog又依賴於貓、貓依賴於魚。。。(當然這裡只是打個比方)

因為我們不知道具體有幾層依賴,所以使用了遞歸的方法,直到將所有依賴的對象得到後再創建實例。

然後我們再來測試

Container container = new Container();
container.ResgisterType<IAnimal, Animal>();
container.ResgisterType<IDog, Dog>();
IAnimal animal= container.Rerolve<IAnimal>();

註意,如果測試標記特性的一定不要忘了在構造函數上標記特性,然後我們會發現最終也可以得到animal對象。

 

然後,創建對象這一塊我們先告一段落。接下來進行生命周期管理。

一般的IOC容器都支持三種類型:Transient每次都得到一個新的對象、Scoped同一個域(或者請求、線程)中使用同一個對象、Singleton整個程式生命周期都使用同一實例對象。

那按照我們以上的代碼怎麼才能實現生命周期管理呢?我是這麼想的:既然創建對象的工作都是由我容器來做了,那麼我們在創建完對象之後能不能像註冊類型一樣將對象保存起來呢?

所以我這裡使用了簡單的字典來存儲對象實例,然後通過判斷使用的哪一種生命周期來返回新的對象或是直接返回字典里的對象。直接改造上面的代碼了:

/// <summary>
        /// 根據提供的類型創建類型實例並返回
        /// </summary>
        /// <param name="type"></param>
        /// <returns></returns>
        private object CreateType(Type type)
        {
            var ctorArray = type.GetConstructors();
            if (ctorArray.Count(c => c.IsDefined(typeof(LInjectionConstructorAttribute), true)) > 0)
            {
                //獲取帶特性標記的構造函數參數
                foreach (var cotr in type.GetConstructors().Where(c => c.IsDefined(typeof(LInjectionConstructorAttribute), true)))
                {
                    var paraArray = cotr.GetParameters();//獲取參數數組
                    if (paraArray.Length == 0)
                    {
                        //return Activator.CreateInstance(type);
                        return GetSocpe(type);
                    }

                    List<object> listPara = new List<object>();
                    foreach (var para in paraArray)
                    {
                        string paraKey = para.ParameterType.FullName;//參數類型名稱
                                                                     //從字典中取出緩存的目標對象並創建對象
                        Type paraTargetType = (Type)ContainerTypeDictionary[paraKey];
                        object oPara = CreateType(paraTargetType);//遞歸
                        listPara.Add(oPara);
                    }
                    //return Activator.CreateInstance(type,listPara.ToArray());
                    return GetSocpe(type, listPara.ToArray());
                }

                return GetSocpe(type);
                //return Activator.CreateInstance(type);
            }
            else
            {
                //沒有標記特性則使用參數最多的構造函數
                var ctor = ctorArray.OrderByDescending(c => c.GetParameters().Length).FirstOrDefault();
                var paraArray = ctor.GetParameters();//獲取參數數組
                if (paraArray.Length == 0)
                {
                    //return Activator.CreateInstance(type);
                    return GetSocpe(type);
                }

                List<object> listPara = new List<object>();
                foreach (var para in paraArray)
                {
                    string paraKey = para.ParameterType.FullName;//參數類型名稱
                                                                 //從字典中取出緩存的目標對象並創建對象
                    Type paraTargetType = (Type)ContainerTypeDictionary[paraKey];
                    object oPara = CreateType(paraTargetType);//遞歸
                    listPara.Add(oPara);
                }
                return GetSocpe(type, listPara.ToArray());
                //return Activator.CreateInstance(type, listPara.ToArray());
            }
        }
        private object GetSocpe(Type type, params object[] listPara)
        {
            if (_scopeType == (int)Scope.Singleton)
            {
                return GetTypeSingleton(type, listPara);
            }
            else if (_scopeType == (int)Scope.Transient)
            {
                return GetTypeTransient(type, listPara);
            }
            else
            {
                return GetTypeScoped(type, listPara);
            }
        }

        #region 生命周期
        /// <summary>
        /// 設置獲取實例對象生命周期為Singleton
        /// </summary>
        /// <param name="type"></param>
        /// <param name="listPara"></param>
        /// <returns></returns>
        private object GetTypeSingleton(Type type, params object[] listPara)
        {
            if (ContainerExampleDictionary.ContainsKey(type.FullName))
            {
                lock (locker)
                {
                    if (ContainerExampleDictionary.ContainsKey(type.FullName))
                    {
                        return ContainerExampleDictionary[type.FullName];
                    }
                }
            }

            if (listPara.Length == 0)
            {
                var Example = Activator.CreateInstance(type);
                ContainerExampleDictionary.Add(type.FullName, Example);
                return Example;
            }
            else
            {
                var Example = Activator.CreateInstance(type, listPara.ToArray());
                ContainerExampleDictionary.Add(type.FullName, Example);
                return Example;
            }
        }

        /// <summary>
        /// 設置獲取實例對象生命周期為Transient
        /// </summary>
        /// <param name="type"></param>
        /// <param name="listPara"></param>
        /// <returns></returns>
        private object GetTypeTransient(Type type, params object[] listPara)
        {
            if (listPara.Length == 0)
            {
                var Example = Activator.CreateInstance(type);
                //ContainerExampleDictionary.Add(type.FullName, Example);
                return Example;
            }
            else
            {
                var Example = Activator.CreateInstance(type, listPara.ToArray());
                //ContainerExampleDictionary.Add(type.FullName, Example);
                return Example;
            }
        }

        /// <summary>
        /// 設置獲取實例對象生命周期為Scoped
        /// </summary>
        /// <param name="type"></param>
        /// <param name="listPara"></param>
        /// <returns></returns>
        private object GetTypeScoped(Type type, params object[] listPara)
        {
            var pid = System.Threading.Thread.CurrentThread.ManagedThreadId;
            if (ContainerExampleDictionary.ContainsKey(type.FullName + pid))
            {
                lock (locker)
                {
                    if (ContainerExampleDictionary.ContainsKey(type.FullName + pid))
                    {
                        return ContainerExampleDictionary[type.FullName + pid];
                    }
                }
            }

            if (listPara.Length == 0)
            {
                var Example = Activator.CreateInstance(type);
                ContainerExampleDictionary.Add(type.FullName + pid, Example);
                return Example;
            }
            else
            {
                var Example = Activator.CreateInstance(type, listPara.ToArray());
                ContainerExampleDictionary.Add(type.FullName + pid, Example);
                return Example;
            }
        }
        #endregion
private static Dictionary<string, object> ContainerExampleDictionary = new Dictionary<string, object>();
        private static int _scopeType;
        private static readonly object locker = new object();
        public int scopeType
        {
            get
            {
                return _scopeType;
            }
            set
            {
                _scopeType = value;
            }
        }
        public enum Scope
        {
            Singleton = 0,
            Transient = 1,
            Scoped = 2
        }

然後調用的時候先聲明下要使用的聲明周期類型就行啦

Container container = new Container();
container.scopeType = (int)Container.Scope.Singleton;
container.ResgisterType<IAnimal, Animal>();
container.ResgisterType<IDog, Dog>();
IAnimal animal= container.Rerolve<IAnimal>();

說下三種生命周期管理的實現:

Transient:則可以直接創建一個實例

Scoped:使用的是同一個線程內使用同一個對象實例,使用var pid = System.Threading.Thread.CurrentThread.ManagedThreadId;獲取線程id來判斷的

Singleton:這種則只需一個單例模式獲取就好了

 

到這裡就先告一段落了,以上只是一個簡單實現,代碼還有需改進的地方以及可以擴展的功能,歡迎提意見指出錯誤。同時代碼已上傳GigHub,還有不懂的可以參考下代碼。

 源碼地址:https://github.com/liangchengxuyuan/IocContainer


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

-Advertisement-
Play Games
更多相關文章
  • 公司業務量越來越大,為了使公司的開發效率有一定的提高,老闆拍板由我牽頭開發了一個快速開發平臺。 我們主要做的是對OA、ERP、ERP等有一定需求的客戶,搭建出一個通用的後臺,再配合一些開發組件,能夠使工作輕鬆不少。 另外,老闆一再強調要對app進行支持,能一次部署到安卓和ios上,微信當然也不能落下 ...
  • 首先創建一個DataGridView控制項,然後創建列(包括列名的定義), 由於我不是和資料庫進行連接,只是為了輸出好看一點。 刪除所有數據: while (this.dataGridView1.Rows.Count != 0) { this.dataGridView1.Rows.RemoveAt(0 ...
  • 在VS中添加bartender的COM組件引用後(一定要添加,否則會提示找不到BarTender.Application): 如是遇到標簽等設置無誤,但仍然無法列印時,有可能存在以下問題: 1.印表機驅動有問題,需重裝驅動; 2.bartender在企業版中需要設置Seagull License S ...
  • 寫個重新載入 ocelot 配置的介面 Intro 我們想把 ocelot 的配置放在自己的存儲中,放在 Redis 或者資料庫中,當修改了 Ocelot 的配置之後希望即時生效,又不想在網關這邊定時刷新 ocelot 配置,ocelot 配置沒變化的時候,定時刷新配置是一種無意義的資源浪費,oce ...
  • 在C#開發應用程式的過程中,圖片一般會存放在文件系統中,當然圖片也可以二進位的方式存放到資料庫中,不過一般不建議存放在資料庫中,因為圖片占用的空間還是挺大的,特殊情況下可以考慮將圖片存在數據。此文將介紹如何將圖片存放在Sqlserver資料庫中,並從資料庫中讀取出圖片信息。 在將圖片存儲到資料庫之前 ...
  • 一. 模型綁定 ASP.NET Core MVC 中的模型綁定,是將 HTTP 請求中的數據映射到action方法參數。 這些參數可能是簡單類型的參數,如字元串、整數或浮點數,也可能是複雜類型的參數。 當 MVC 收到 HTTP 請求時,它會將此請求路由定位到控制器的指定action方法。預設路由模 ...
  • 分析及思路 來看一下項目目錄結構 炒雞正常的三板斧src+docs+tests。咦,怎麼會多出一個build的文件夾呢,這就是我們今天要研究的目錄。今天我會帶著大家在五分鐘之內編寫一個極簡的編譯腳本。 build內及其簡單 思路很明顯了:獲取當前腳本所在目錄=>獲取上一級目錄=>回到上一級(.sln ...
  • 前臺: <title>標題</title> <link href="EasyUi_v1.3.4/easyui/themes/default/easyui.css" rel="stylesheet" /> <link href="EasyUi_v1.3.4/easyui/themes/icon.css ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...