在java的spring中有自動注入功能,使得代碼變得更加簡潔靈活,所以想把這個功能移植到c#中,接下來逐步分析實現過程
1.使用自動注入場景分析
在asp.net mvc中," />

ASP.NET MVC實現依賴注入的完整過程

 更新時間:2020-01-15 16:00:32   作者:佚名   我要評論(0)

前言
在java的spring中有自動注入功能,使得代碼變得更加簡潔靈活,所以想把這個功能移植到c#中,接下來逐步分析實現過程
1.使用自動注入場景分析
在asp.net mvc中,

前言

在java的spring中有自動注入功能,使得代碼變得更加簡潔靈活,所以想把這個功能移植到c#中,接下來逐步分析實現過程

1.使用自動注入場景分析

在asp.net mvc中,無論是什么代碼邏輯分層,最終的表現層為Controller層,所以我們注入點就是在Controller中,這里我們需要替換默認的ControllerFactory,掃描代碼中標記需要注入的對象,進行實例化注入

public class FastControllerFactory : DefaultControllerFactory
  {
    public override IController CreateController(RequestContext requestContext, string controllerName)
    {
      Type type = this.GetControllerType(requestContext, controllerName);
      Object obj = GetControllerInstance(requestContext, type);

      //Controller中標記AutoWired屬性的自動注入
      List<FieldInfo> AutoWiredFieldList = type.GetRuntimeFields().Where(f => f.GetCustomAttribute(typeof(AutoWired)) != null).ToList();
      foreach (FieldInfo field in AutoWiredFieldList)
      {
        field.SetValue(obj, InjectUtil.Container.Resolve(field.FieldType));
      }
      return obj as IController;
    }
  }

FastControllerFactory就是我們自定義的一個Controller工廠,重寫CreateController方法,對標記了AutoWired這個自定義注解的變量,從Bean容器中取出實例進行賦值,同時我們還需要在Global文件中的Start方法中,進行默認工廠進行替換

ControllerBuilder.Current.SetControllerFactory(new FastControllerFactory());

2.IOC容器的實現

c#中的自定義容器有很多開源成熟的框架,例如AutoFac等,這里我們是自己實現一個輕量級的版本

源碼地址:https://gitee.com/grassprogramming/FastIOC

這里就重點說一下如何在asp.net mvc中的使用,首先我們需要對需要注入的Bean對象進行標記,這個標記就叫做Component,

在asp.net mvc Global文件中的Start方法中,我們需要將整個項目中需要自動注入的Bean加入到容器中

public class InjectUtil
  {
    public static ContainerBuilder Container;
    public static void Init()
    {
      Container = new ContainerBuilder();
       //獲取所有程序集
      var assemblies = System.Web.Compilation.BuildManager.GetReferencedAssemblies().Cast<Assembly>().ToArray();
      //注入所有Component組件
      Container.RegisterAssemblyTypes(assemblies, typeof(Component),true);
      Container.Build();
    }
  }

到這里Controller層面的事項就已經完成了,接下來就需要在IOC容器中初始化Bean實例方法中進一步處理

private Object GetInstance(RegisterEntity Entity)
    {
      Object obj = null;
      if (Entity.IsEnableIntercept)
      {
        bool IsExtend = Entity.RealType == Entity.RegistType;
        obj = DynamictProxy.CreateProxyObject(Entity.RealType, Entity.RegistType, Entity.InterceptType, IsExtend, Entity.IsInterceptAllMethod);


      }
      else
      {
        var constructors = Entity.RegistType.GetConstructors();
        obj = constructors[0].Invoke(new Object[] { });
      }
      //這里使用單例模式將實例化Instance存儲,提前暴露未進行后續設置的對象實例
      if (!SingleInstanceDic.ContainsKey(Entity.RealType))
      {
        SingleInstanceDic.Add(Entity.RealType, obj);
      }
    
      //如果這個class標記了Component,且有標記了AutoWired的Field,進行自動注入
      if (Entity.RealType.GetCustomAttribute(typeof(Component), true) != null)
      {
        //這里要使用GetRuntimeFields,此方法返回在指定類型上定義的所有字段,包括繼承,非公共,實例和靜態字段。
        foreach (FieldInfo Field in Entity.RealType.GetRuntimeFields())
        {
          if (Field.GetCustomAttribute(typeof(AutoWired), true) != null)
          {
            Type FieldType = Field.FieldType;
            if (Contains(FieldType))
            {
              //判斷單例存儲中是否包含,如果有,取出賦值,這里可以防止循環依賴導致的死循環
              if (SingleInstanceDic.ContainsKey(FieldType))
              {
                Field.SetValue(obj, SingleInstanceDic[FieldType]);
              }
              else
              {
                Field.SetValue(obj, Resolve(FieldType));
              }
              
            }
          }
        }
      }
      return obj;

    }

GetInstance方法就是實例化Bean對象的核心方法,其實很簡單,就是通過反射創建對象,其中需要注意的有兩點

1)對于一個Bean初始化時需要掃描Bean中的所有變量,如果內部還有依賴注入的嵌套對象,需要使用遞歸,直到沒有需要注入的Field

2)我這里使用的是單例模式,因為在測試過程中可能存在在A類中對B進行依賴注入,在B類中對A進行依賴注入,常規創建過程,如果使用遞歸進行掃描,就會進入死循環,內存溢出,所以使用對象的單例,一旦創建就放入字典中,如果再次掃描到該對象需要注入,則直接取出使用,就避免了循環引用

3.其他

對其他不在Controller中使用的類需要依賴注入,則需要直接從IOC的Bean容器取出使用

 private AuthUtil @AuthUtil = InjectUtil.Container.Resolve<AuthUtil>();

功能到這里就全部分析完畢了,最后打個廣告,自己寫的ASP.NET MVC快速開發框架,希望支持一波

地址:https://gitee.com/grassprogramming/FastExecutor

總結

以上就是這篇文章的全部內容了,希望本文的內容對大家的學習或者工作具有一定的參考學習價值,謝謝大家對腳本之家的支持。

您可能感興趣的文章:

  • 解讀ASP.NET 5 & MVC6系列教程(7):依賴注入
  • 擴展ASP.NET MVC三層框架且使用StructureMap實現依賴注入1-Model層

相關文章

  • ASP.NET MVC實現依賴注入的完整過程

    ASP.NET MVC實現依賴注入的完整過程

    前言 在java的spring中有自動注入功能,使得代碼變得更加簡潔靈活,所以想把這個功能移植到c#中,接下來逐步分析實現過程 1.使用自動注入場景分析 在asp.net mvc中,
    2020-01-15
  • .NET資源泄露與處理方案知識點分享

    .NET資源泄露與處理方案知識點分享

    .NET雖然擁有強大易用的垃圾回收機制,但并不是因為這樣,你就可以對資源管理放任不管,其實在稍不注意的時候,可能就造成了資源泄露,甚至因此導致系統崩潰,到那時
    2020-01-15
  • asp.net大文件上傳解決方案實例代碼

    asp.net大文件上傳解決方案實例代碼

    以ASP.NET Core WebAPI 作后端 API ,用 Vue 構建前端頁面,用 Axios 從前端訪問后端 API ,包括文件的上傳和下載。 準備文件上傳的API #region 文件上傳 可以帶參
    2020-01-15
  • .Net Core讀取Json配置文件的實現示例

    .Net Core讀取Json配置文件的實現示例

    前言:在與傳統的asp.net MVC項目相比,.net core項目在項目目錄的文件結構上和功能上與前者都有很大的區別。例如:在.net core中使用Startup.cs取代Global.asax文件
    2020-01-15
  • .net core部署到windows服務上的完整步驟

    .net core部署到windows服務上的完整步驟

    前言 Net core 項目部門在Windows有很多種方式,大致有以下幾種, dotnet 命令, iis(windowshosts), 一些開源的應用容器(docker ) 基于一些exe 程
    2020-01-15
  • .NET Core 微信小程序退款步驟——(統一退款)

    .NET Core 微信小程序退款步驟——(統一退款)

    繼上一篇".NET Core 微信小程序支付——(統一下單)后",本文將實現統一退款功能,能支付就應該能退款嘛,一般涉及到錢的東西都會比較敏感,所以在設計退款流程時一
    2020-01-15
  • .NET core 3.0如何使用Jwt保護api詳解

    .NET core 3.0如何使用Jwt保護api詳解

    摘要: 本文演示如何向有效用戶提供jwt,以及如何在webapi中使用該token通過JwtBearerMiddleware中間件對用戶進行身份認證。 認證和授權區別? 首先我們要弄清楚認證
    2020-01-15
  • .Net Core下HTTP請求IHttpClientFactory示例詳解

    .Net Core下HTTP請求IHttpClientFactory示例詳解

    使用方式 IHttpClientFactory有四種模式: 基本用法 命名客戶端 類型化客戶端 生成的客戶端 基本用法 在 Startup.ConfigureServices 方法中,通過在
    2020-01-15
  • .net core webapi通過中間件獲取請求和響應內容的方法

    .net core webapi通過中間件獲取請求和響應內容的方法

    本文主要根據中間件來實現對.net core webapi中產生的請求和響應數據進行獲取并存入日志文件中; 這里不詳細介紹日志文件的使用。你可以自己接入NLog,log4net,Exce
    2020-01-15
  • VS2019以及MFC的安裝詳細教程

    VS2019以及MFC的安裝詳細教程

    本文為大家分享了VS2019以及MFC的安裝詳細教程,供大家參考,具體內容如下 一、安裝過程: 1、搜索visual studio 進入官網。 2、選擇community 2019下載。 3、現在
    2020-01-15

最新評論

买宝宝用品赚钱吗