較複雜的應用程式都是由多個項目組織成的,項目可以劃分成程式集(Assemblies)和宿主(Hosts),也就是應用程式的入口。 Assemblies 通常是常見的類庫項目,包括可以重用的功能以方便測試,通常包括下麵的組件: Views, Controllers 和 Models; 服務; 持久類 ...
較複雜的應用程式都是由多個項目組織成的,項目可以劃分成程式集(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的偽代碼類似於:
1 var builder = new ContainerBuilder();
2 builder.Register(new ConfigurationSettingsReader());
3 using (var container = builder.Build())
4 {
5 var shell = container.Resolve<Shell>();
6 shell.Execute();
7 }
上述代碼中new ConfigurationSettingsReader()就是autofac從配置文件中讀取相關的組件配置,一般使用XML文件進行配置,autofac的xml配置文檔可以看XmlConfiguration,使用配置文件也有缺點:
- 不是強類型的,編譯器無法發現錯誤,沒有智能提示;
- 配置文件會變得越來越大;
- 維護多個配置文件比較困難;
- 文件不適合用於複雜的環境;
上述缺點我們可以通過.NET代碼塊封裝相關的配置細節,在XML文件中只保留粗粒度的配置,Autofac可以通過Module進行配置塊的封裝,具體可以參考文檔StructuringWithModules。
我這裡取個例子:
1 public class LoggingModule : Module
2 {
3 public Mode Mode { get; set; }
4 public static string EventLogName = "網站通行證";
5 public static string EventLogSource = "應用程式";
6
7 public LoggingModule()
8 {
9 Mode = NCASService.Mode.Diagnostics;
10 }
11
12 protected override void Load(ContainerBuilder builder)
13 {
14 // configure logging
15 var logger = GetLoggerForWindows(Mode);
16 builder.RegisterInstance(logger);
17 builder.RegisterInstance(logger.Get("DefaultLog"));
18 base.Load(builder);
19 }
20
21 static INamedProvider<ILog> GetLoggerForWindows(Mode mode)
22 { // configuring different logging based on our mode
23 switch (mode)
24 {
25 case Mode.Release:
26 // write all informational and higher events to indows event log
27 LoggingStack.UseEventLog(EventLogName, EventLogSource)
28 .Filter(LogLevel.Info, LogLevel.Max);
29 // dump all warning and higher messages to rolling text log
30 LoggingStack.UseRollingLog(@"logs\errorlog.txt", 100.Kb(), 10)
31 .Filter(LogLevel.Warn, LogLevel.Fatal);
32 break;
33 case Mode.Diagnostics:
34 // dump all messages to daily log
35 LoggingStack.UseDailyLog(@"log.txt");
36 break;
37 case Mode.Debug:
38 // Visual studio would get these messages
39 return TraceLog.Provider;
40 default:
41 throw new ArgumentOutOfRangeException("mode");
42 }
43 return LoggingStack.GetLogProvider();
44 }
上述是把我們的日誌模塊的配置用代碼進行配置,我們的XML配置文件中的配置就會變得很簡單:
1 <!-- Production configuration -->
2 <module type="NdonFramework.NCASService.LoggingModule, NCASService">
3 </module>
4
5 <!-- Development configuration -->
6 <module type="NdonFramework.NCASService.LoggingModule, NCASService">
7 <properties>
8 <property name="Mode" value="Debug" />
9
10 </properties>
11 </module>
12
13 <!-- Sandbox configuration -->
14 <module type="NdonFramework.NCASService.LoggingModule, NCASService">
15 <properties>
16 <property name="Mode" value="Diagnostics" />
17 </properties>
18 </module>
使用模塊組織不同程式集中的組件註冊到容器里,模塊我一般需要配置以下內容:
- 配置日誌記錄並註冊ILog組件(例如記錄到控制台,文本文件、Windows日誌文件)
- 配置異常處理策略
- 註冊數據訪問類
- 註冊交叉關註點
- 配置驗證規則
通過Autofac的Module分解項目組件間的複雜關係。
參考鏈接:http://www.cnblogs.com/shanyou/archive/2010/05/28/1746711.html