理解為什么要使用Ioc


轉自 http://www.cnblogs.com/shanyou/archive/2010/05/28/1746711.html

較復雜的應用程序都是由多個項目組織成的,項目可以划分成程序集(Assemblies)和宿主(Hosts),也就是應用程序的入口。

Assemblies 通常是常見的類庫項目,包括可以重用的功能和方便測試,通常包括下面的組件:

  • Views, Controllers 和 Models
  • 服務
  • 持久類 和 repositories
  • Decorators
  • Reusable user controls
  • 規則庫
  • 業務邏輯

這些項目通常不應該直接依賴於下面的組件:

  • IoC 容器程序集;
  • 日志記錄框架 ;
  • 數據訪問框架;
  • 其他第三方類庫.

為了分離這些邏輯,我們可以定義一些接口,然后通過配置代碼將具體實現關聯起來,例如日志記錄我們可以定義一個接口ILog,生產環境下我們可以把它改成用Apache log4net或者企業類庫的日志記錄模塊都可以。由於這是接口定義和實現分離的,我們可以在不同環境下使用不同的實現,只需要通過配置修改就可以而不要重新編譯代碼。

Hosts代表應用程的入口,有下面這些形式:

  • 桌面應用程序:
    • Windows.Forms;
    • WPF;
  • 控制台應用程序;
  • windows 服務;
  • Web應用程序
  • Microsoft Office Add-Ins;
  • Microsoft Azure Roles.

Host負責構建應用程環境(上下文),並把它傳遞給應用程序的入口,在IOC容器方面來說,通過配置容器中的應用程序組件,獲取Shell類並運行。通常Host項目都很小,主要完成兩個方面的工作:配置容器和調用Shell.Run()。

用Autofac的Host的偽代碼類似於

var builder = new ContainerBuilder();
builder.Register(new ConfigurationSettingsReader());
using (var container = builder.Build())
{
   var shell = container.Resolve<Shell>();
   shell.Execute();
}

上述代碼中new ConfigurationSettingsReader()就是autofac從配置文件中讀取相關的組件配置,一般使用XML文件進行配置,autofac的xml配置文檔可以看XmlConfiguration,使用配置文件也有缺點:

  • 不是強類型的,編譯器無法發現錯誤,沒有智能提示
  • 配置文件會變得越來越大
  • 維護多個配置文件比較困難
  • 文件文件不適合用於復雜的環境

上述缺點我們可以通過.NET代碼塊封裝相關的配置細節,在XML文件中只保留粗粒度的配置,Autofac可以通過Module進行配置塊的封裝,具體可以參考文檔StructuringWithModules

我這里取個例子:

public  class LoggingModule : Module
   {
       public Mode Mode { get; set; }
       public static string EventLogName = "網站通行證";
       public static string EventLogSource = "應用程序";

       public LoggingModule()
       {
           Mode = NCASService.Mode.Diagnostics;
       }

       protected override void Load(ContainerBuilder builder)
       {
           // configure logging 
           var logger = GetLoggerForWindows(Mode);   
           builder.RegisterInstance(logger);  
           builder.RegisterInstance(logger.Get("DefaultLog"));
           base.Load(builder);
       }

       static INamedProvider<ILog> GetLoggerForWindows(Mode mode)
       {  // configuring different logging based on our mode
           switch (mode)
           {
               case Mode.Release:
                   // write all informational and higher events to indows event log
                   LoggingStack.UseEventLog(EventLogName, EventLogSource)
                       .Filter(LogLevel.Info, LogLevel.Max);
                   // dump all warning and higher messages to rolling text log  
                   LoggingStack.UseRollingLog(@"logs\errorlog.txt", 100.Kb(), 10)
                       .Filter(LogLevel.Warn, LogLevel.Fatal);
                   break;
               case Mode.Diagnostics:
                   // dump all messages to daily log
                   LoggingStack.UseDailyLog(@"log.txt");
                   break;
               case Mode.Debug:
                   // Visual studio would get these messages 
                   return TraceLog.Provider;
               default:
                   throw new ArgumentOutOfRangeException("mode");
           }
           return LoggingStack.GetLogProvider();
       }

上述是把我們的日志模塊的配置用代碼進行配置,我們的XML配置文件中的配置就會變得很簡單:

<!-- Production configuration -->
<module type="NdonFramework.NCASService.LoggingModule, NCASService">
</module>

<!-- Development configuration -->
<module type="NdonFramework.NCASService.LoggingModule, NCASService">
  <properties>
    <property name="Mode" value="Debug" />

  </properties>
</module> 

<!-- Sandbox configuration -->
<module type="NdonFramework.NCASService.LoggingModule, NCASService">
  <properties>
    <property name="Mode" value="Diagnostics" />
  </properties>
</module>

使用模塊組織不同程序集中的組件注冊到容器里,模塊我一般需要配置以下內容:

  • 配置日志記錄並注冊ILog組件(例如記錄到控制台,文本文件、Windows日志文件)
  • 配置異常處理策略
  • 注冊數據訪問類
  • 注冊交叉關注點
  • 配置驗證規則

通過Autofac的Module分解項目組件間的復雜關系。


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM