一、Autofac入门
想要将autofac集成到你的应用程序中需要经过如下步骤:
1、使用控制翻转(IoC)的思想架构你的应用程序;
2、添加autofac引用;
3、在应用程序入口...(At application startup...);
4、创建一个ContainerBuilder对象;
5、注册组件;
6、为以后的使用,生成并保存容器;
7、在应用执行期间...(During application execution...);
8、从容器中创建一个生命周期域;
9、使用生命周期域解析(resolve)组件实例。
这个入门指导将带领你通过一个简单的控制台应用程序来完成以上这些步骤。一旦你掌握了这些基础知识,就可以进一步学习更高级的概念以及如何将autofac集成到WCF、ASP.NET或其他类型的应用程序中。
1.1、构建应用(Structuring the Application)
控制反转背后的思想是,与其将应用中的类捆绑到一起从而提升相互间的依赖关系,不如在类被构造时再通过注入的方式引入相关的依赖。如果你想更深入了解控制反转,马丁福勒有一篇优秀的文章解释了依赖注入/控制反转。
在我们的控制台应用程序示例中,我将定义一个输出当前日期的类。但是我们并不想将他直接与控制台绑定,因为我们可能希望在今后能够对这个类进行测试或者将该类应用到控制台意外的地方。
我会尽可能的对输出日期的方式进行抽象,以便于今后我们能够使用其他的版本替换他。
我将会这样设计我们的类:
namespace DemoApp
{
//接口帮助我们将“输出"从控制台类解耦,
//我们并不关心输出操作是如何执行的,只要能输出就够了。
public interface IOutput
{
void Write(string content);
} //ConsoleOuput实现了IOutput接口,
//他将信息写到了控制台。从技术上来说我们还可以通过
//实现IOutput接口,将信息写入跟踪调试或者其他地方
public class ConsoleOutput : IOutput
{
public void Write(string content)
{
Console.WriteLine(content);
}
} //这个接口解耦了输入日期的机制,与IOutput一样,实际的处理机制
//通过接口进行了抽象
public interface IDateWriter
{
void WriteDate();
} //注意TodayWriter定义了一个使用IOutput作为形参的构造函数,
//这样做可以根据传入的具体具体类型将日期写到任何地方,他实现了
//WriteDate()方法来输入日期,你可以让他以不同的格式输出到
//不同的地方
public class TodayWriter : IDateWriter
{
private IOutput _output;
public TodayWriter(IOutput output)
{
this._output = output;
}
public void WriteDate()
{
this._output.Write(DateTime.Today.ToShortDateString());
}
}
}
现在我们有一个合理的结构来设置依赖关系,那就让我们通过autofac来将他们结合到一起吧!
1.2、添加autofac引用
第一步是将autofac添加到你的应用。在这个应用中我们只是用core autofac(这里翻译成核心autofac感觉怪怪的,直接上英文还显得明了一些),其他的应用类型可以使用额外的Autofac integration libraries。
最简单的方式是通过NuGet来添加autofac引用,”Autofac"包已经包含了你需要的所有核心功能。
1.3、应用程序入口
在应用程序入口你需要创建一个ContainerBuilder,并将组件注册到其中。组件可能是一个表达式、.net类型或者其他的暴露了一个或多个服务并能够被其他依赖项所使用的代码。简单来说,可以将组件理解为实现了某个接口的.net类型,列如:
public class SomeType : IService
{
}
你可以通过两种方式来识别它:一种方式可以将其看作本身的类型(SomeType),另一种可以将其视为接口(IService)。在这种情况下组件就是SomeType,而其所暴露的服务就是SomeType和IService。
在autofac中,你需要将其注册到ContainerBuilder中,比如:
//创建ContainerBuilder对象
var builder = new ContainerBuilder(); //通常情况下你所感兴趣的是组件通过接口暴露的服务
builder.RegisterType<SomeType>().As<IService>(); //然而如果你同时也对组件通过自身类型暴露的服务感兴趣(并不常见),可以如下方式进行注册:
builder.RegisterType<SomeType>().AsSelf().As<IService>();
在我们的应用中,需要对我们所有的组件(class)进行注册,并暴露他们所提供的服务(Interface)。
我们同样需要保存我们容器,以便于之后通过容器来解析类型。
using System;
using Autofac; namespace DemoApp
{
public class Program
{
private static IContainer Container{ get; set; }
static void Main(string[] args)
{
var builder = new ContainerBuilder();
builder.RegisterType<ConsoleOutput>().As<IOutput>();
builder.RegisterType<TodayWrite>().As<IDateWrite>();
Container = builder.Build(); //我们将在WriteDate方法中使用依赖注入,该方法我们稍后实现。
WriteDate();
}
}
}
现在,我们已经拥有了容器,注册了组件并暴露了适当的服务,现在就让我们来使用它。
1.4、应用的执行
在应用执行时,你需要使用注册的组件。为此,将从一个生命周期域中解析组件。
容器本身是一个生命周期域,技术上来说,你可以直接通过容器进行解析。然后这样的做法是不被推荐的。
当你解析一个组件,根据你定义的实例范围,一个新的对象实例将被创建。(解析一个组件大致等效于“new”一个类的实例,这是最最简化的一种说法,但从比喻的角度来看却是很恰当的。)一些组件可能需要被销毁(比如实现了IDisposable接口的组件),当生命周期域被释放的时候autofac能够帮你处理这些组件。
然而,container存在于你的应用程序生命周期中,如果你直接从容器中解析了很多的组件,最终将会有很多的组件实例等待销毁。这很糟糕(这可能会导致内存溢出)。
相反,从container中创建子生命周期域,并从子生命周期域中解析组件,那么当子生命周期域结束时所有从该域中创建的组件都将被销毁。(当你使用Autofac integration libraries时,创建子生命周期域的任务将由autofac完成,你不必过多考虑)
在我们的例子中,我将实现WriteDate方法,在一个域中执行输出,并在完成后销毁该域。
namespace DemoApp
{
public class Program
{
private IContainer Container{ get; set; } static void Main(string[] args)
{
// ...the stuff you saw earlier...
} public static void WriteDate()
{
//创建域,解析IDateWriter
using(var scope = Container.BeginLifetimeScope())
{
var writer = scope.Resolve<IDateWriter>();
writer.WriterDate();
}
}
}
}
现在当你运行程序时:
1、“WriteDate”方法向autofac请求一个IDateWtiter;
2、autofac发现IDateWtiter映射到TodayWriter,然后创建TodayWriter对象;
3、autofac发现TodayWriter的构造函数需要一个IOutput参数;
4、autofac发现IOutput映射到ConsoleOutput,然后创建ConsoleOutput实例;
5、autofac使用ConsoleOutput实例完成TodayWriter的创建工作;
6、autofac完成TodayWriter创建,并将TodayWriter实例返回给“WriteDate”。
以后,如果你希望你的应用程序输出不同的日期,你可以实现不同的IDateWriter并修改应用程序入口处的注册,这样你就不必在修改其他的类。耶,控制反转!!
注意:一般来说,服务定位很大程度上被视为一种“反模式”,也就是说到处都通过手动创建域(scope)以及直接使用container并不是最好的选择。使用Autofac integration libraries通常不需要我们去手动的完成示例应用程序中创建域以及解析组件的工作,在应用程序中“顶层”定位以及手动解析组件是非常罕见的,当然,如何设计你的应用程序取决于你自己。
PS:本系列博客是对autofac英文资料的翻译,主要目的是为了提高自己英文阅读能力,同时能够帮助有需要的人,原文地址http://autofac.readthedocs.org/en/latest/getting-started/index.html。翻译过程中我将“resolve”译为“解析”,但在阅读中感觉意思并未通达,绞尽脑汁仍未找到恰当的词语代替,若各位园友有合适翻译还望指教。
一、Autofac入门的更多相关文章
- Autofac 入门
Autofac 入门文档 原文链接:http://docs.autofac.org/en/latest/getting-started/index.html 在程序中使用Autofac的基本模式是: ...
- Autofac入门
注意:本文为原创文章,任何形式的转载.引用(包括但不限于以上形式)等,须先征得作者同意,否则一切后果自负. 简介 Autofac 是一个令人着迷的.NET IoC 容器. 它管理类之间的依赖关系.当应 ...
- [翻译] Autofac 入门文档
原文链接:http://docs.autofac.org/en/latest/getting-started/index.html 在程序中使用Autofac的基本模式是: 用控制反转(IoC)的思想 ...
- .NET手记-Autofac入门Getting Started
内容主要翻译自官方文档,原文请看:http://autofac.readthedocs.org/en/latest/getting-started/index.html#application-sta ...
- [转]ASP.NET MVC IOC 之AutoFac攻略
本文转自:http://www.cnblogs.com/WeiGe/p/3871451.html 一.为什么使用AutoFac? 之前介绍了Unity和Ninject两个IOC容器,但是发现园子里用A ...
- ASP.NET MVC IOC 之AutoFac攻略
一.为什么使用AutoFac? 之前介绍了Unity和Ninject两个IOC容器,但是发现园子里用AutoFac的貌似更为普遍,于是捯饬了两天,发现这个东东确实是个高大上的IOC容器~ Autofa ...
- 不复杂的Autofac注入
private static void SetAutofacWebAPI() { var builder = new ContainerBuilder(); #region 配置注册方法 string ...
- Autofac 一个使用Demo
一:接口 二:实现: 三:调用: 首先上图: 一:接口代码 public interface IPersonDa { PersonEntity Get(int id); } 二:实现 public c ...
- ASP.NET MVC IOC 之AutoFac
ASP.NET MVC IOC 之AutoFac攻略 一.为什么使用AutoFac? 之前介绍了Unity和Ninject两个IOC容器,但是发现园子里用AutoFac的貌似更为普遍,于是捯饬了两天, ...
随机推荐
- UITableView 属性集合
UITableView-------表视图--继承UIScrollView并遵守NSCoding协议 属性 frame-------------设置控件的位置和大小 backgroundColor-- ...
- Servlet编码和解码
1.request.setCharacterencoding("XXX"); 前提是POST提交 在客户端编码对value的值进行编码之前,通知客户端用什么码表(XXX)编码 2. ...
- centos mysql 数据存储目录安装位置
rpm -ql mysql查看安装位置 MYSQL默认的数据文件存储目录为/var/lib/mysql.假如要把目录移到/home/data下需要进行下面几步: 1.home目录下建立data目录 c ...
- iOS学习之视图加载过程中会触发的方法(loadView/viewDidLoad/didReceiveMemoryWarning)
1.loadView 这是视图控制器用来加载根视图的方法; 如果需要将自定义的视图作为根视图,则不需要调用父类对该方法的实现([super loadView]);直接将自定义视图通过self.view ...
- C语言--位运算符
一.位运算符 1.按位与:& 1> 功能 * 只有对应的两个二进制位为1时,结果位才为1,否则为0 * 举例:10用二进制表示为1010, 7用二进制表示为0111.对两个数值进行&a ...
- spring boot之使用springfox swagger展示restful的api doc
摘要 springfox swagger展示restful的api doc, swagger is A POWERFUL INTERFACE TO YOUR API. 新增文件: import org ...
- UML--核心元素之包
包是一种容器,如同文件夹一样. 包是UML非常常用的一个元素,它最主要的作用就是容纳并为其他元素分类.包可以容纳用例.业务实体.类图等,也包含子包. 分包的原则 1.如果将元素分为三个包A.B.C,那 ...
- 总结spring下配置dbcp,c3p0,proxool数据源链接池
转载自 http://hiok.blog.sohu.com/66253191.html applicationContext-datasource-jdbc.xml <?xml version= ...
- bzoj1671 [Usaco2005 Dec]Knights of Ni 骑士
Description Bessie is in Camelot and has encountered a sticky situation: she needs to pass through t ...
- 40个UI设计工具
摘要:用户界面设计在持续的基础上不断成长和演变.要跟上时代,你需要关注趋势.新资源和正被实施和谈论的新技术. 导读:用户界面设计在持续的基础上不断成长和演变.要跟上时代,你需要关注趋势.新资源和正被实 ...