使用AutoFac组织多项目应用程序
较复杂的应用程序都是由多个项目组织成的,项目可以划分成程序集(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
推荐阅读
-
基于maven使用IDEA创建多模块项目
-
从零开始搭建前后端分离的NetCore2.2(EF Core CodeFirst+Autofac)+Vue的项目框架之七使用JWT生成Token(个人见解)
-
使用Visual Studio2019创建C#项目(窗体应用程序、控制台应用程序、Web应用程序)
-
从零开始搭建前后端分离的NetCore2.2(EF Core CodeFirst+Autofac)+Vue的项目框架之十一Swagger使用一
-
详解如何使用webpack打包多页jquery项目
-
NET Core 3.0 项目中使用 AutoFac
-
从零开始搭建前后端分离的NetCore2.2(EF Core CodeFirst+Autofac)+Vue的项目框架之六使用过滤器进行全局请求数据验证
-
从零开始搭建前后端分离的NetCore2.2(EF Core CodeFirst+Autofac)+Vue的项目框架之八MemoryCache与redis缓存的使用
-
使用vs2019发布项目“从与我的应用程序相同位置下载系统组件”报错
-
IDEA使用svn拉取多模块项目