IOC容器之Autofac
讲到IOC容器,就必须提到DIP(依赖倒置原则)。
DIP是OOD(面向对象设计)的一个重要思想,在该原则上引申出了\IOC(控制反转)\DI(依赖注入)\IOC容器。
居然讲到IOC容器,当然我们要先了解DIP。
我们来看下定义:
依赖倒置原则(DIP):一种软件架构设计的原则(抽象概念)。
控制反转(IoC):一种反转流、依赖和接口的方式(DIP的具体实现方式)。
依赖注入(DI):IoC的一种实现方式,用来反转依赖(IoC的具体实现方式)。
IoC容器:依赖注入的框架,用来映射依赖,管理对象创建和生存周期(DI框架)。
DIP依赖倒置原则
什么是原则?
答:代表性则问题性的一个定点词 行事所依据的准则
DIP仅仅只是代表依赖倒置的一个依据准则而已。DIP是一种 软件设计原则,它仅仅告诉你两个模块之间应该如何依赖,但是它并没有告诉如何做。
DIP倡导高层模块定义了接口,将不再直接依赖于低层模块,低层模块负责实现高层模块定义的接口。这样,当有新的低层模块实现时,不需要修改高层模块的代码。
由此,我们可以总结出使用DIP的优点:
- 可以修改一部分代码而不影响其他模块。
- 可以修改一部分代码而不会让系统崩溃。
- 组件松耦合,且可复用,提高开发效率。
博客园里很多人用了ATM机的生活例子来进行讲解,ATM机为高层,底层为银行卡,DIP的准则告诉我们ATM机来定义接口然后让银行卡来实现。这样的好处ATM机可以支持很多种银行卡。
IOC控制反转
IOC是什么呢?
IoC则是一种 软件设计模式,它告诉你应该如何做,来解除相互依赖模块的耦合。
控制反转(IoC),它为相互依赖的组件提供抽象,将依赖(低层模块)对象的获得交给第三方(系统)来控制,即依赖对象不在被依赖模块的类中直接通过new来获取。
IoC有两种方式:依赖注入和服务定位
这里就讲到依赖注入DI
DI依赖注入
DI就是将依赖对象的创建和绑定转移到被依赖对象类的外部来实现,将需要依赖(低层模块)对象的引用传递给被依赖(高层模块)对象。
注入的方式有构造函数、属性、接口。
我们来看下构造函数注入例子:
先定义一个数据库操作方法接口
- public interface ILinkmethod
{
void Add();
}
新增一个实现类
public class SqlServerDal: ILinkmethod
{
public void Add()
{
Console.WriteLine("在数据库中添加一条订单!");
}
}
public class ActionOrder
{
private ILinkmethod _ida;//定义一个私有变量保存抽象
//构造函数注入
public Order(ILinkmethod ida)
{
_ida = ida;//传递依赖
}
public void Add()
{
_ida.Add();
}
}
调用:
static void Main(string[] args)
{
SqlServerDal dal = new SqlServerDal();//在外部创建依赖对象
Order order = new Order(dal);//通过构造函数注入依赖
order.Add();
Console.Read();
}
上面的基础铺垫后终于到我们的主题了:
IOC容器
IoC容器实际上是一个DI框架, 它能简化我们的工作量。它包含以下几个功能:
动态创建、注入依赖对象。
管理对象生命周期。
映射依赖关系。
比较流行的Ioc容器:
- Ninject: http://www.ninject.org/
2. Castle Windsor: http://www.castleproject.org/container/index.html
3. Autofac: http://code.google.com/p/autofac/
4. StructureMap: http://docs.structuremap.net/
5. Unity: http://unity.codeplex.com/
注:根据园友 徐少侠 的提醒,MEF不应该是IoC容器。我又查阅了一些资料,觉得MEF作为IoC容器是有点勉强,它的主要作用还是用于应用程序扩展,避免生成脆弱的硬依赖项。
6. MEF: http://msdn.microsoft.com/zh-cn/library/dd460648.aspx
另外,园友 aixuexi 提出Spring.NET也是比较流行的IoC容器。
7. Spring.NET: http://www.springframework.net/
园友 wdwwtzy 也推荐了一个不错的IoC容器:
8. LightInject: http://www.lightinject.net/ (推荐使用Chrome浏览器访问)
以上IOC容器摘抄于木小楠整理。
我们下面来中间讲解下Autofac,为了让大家更加容易入手,请先了解上述内容
Autofac是一款IOC框架,比较于其他的IOC框架,如Spring.NET,Unity,Castle等等所包含的,它很轻量级性能上非常高。
我们来看下官方解释:
Adding Components
Components are registered with a ContainerBuilder:
var builder = new ContainerBuilder();
Autofac can use a Linq expression, a .NET type, or a pre-built instance as a component:
builder.Register(c => new TaskController(c.Resolve<ITaskRepository>()));
builder.RegisterType<TaskController>();
builder.RegisterInstance(new TaskController());
Or, Autofac can find and register the component types in an assembly:
builder.RegisterAssemblyTypes(controllerAssembly);
Calling Build() creates a container:
var container = builder.Build();
To retrieve a component instance from a container, a service is requested. By default, components provide their concrete type as a service:
var taskController = container.Resolve<TaskController>();
To specify that the component's service is an interface, the As() method is used at registration time:
builder.RegisterType<TaskController>().As<IController>();
// enabling
var taskController = container.Resolve<IController>();
方法1:
var builder = new ContainerBuilder();
builder.RegisterType<TestService>();
builder.RegisterType<TestDao>().As<ITestDao>();
return builder.Build();
方法二:
为了统一管理 IoC 相关的代码,并避免在底层类库中到处引用 Autofac 这个第三方组件,定义了一个专门用于管理需要依赖注入的接口与实现类的空接口 IDependency:
/// 依赖注入接口,表示该接口的实现类将自动注册到IoC容器中
这个接口没有任何方法,不会对系统的业务逻辑造成污染,所有需要进行依赖注入的接口,都要继承这个空接口,例如:
public interface IUnitOfWork : IDependency
Autofac 是支持批量子类注册的,有了 IDependency 这个基接口,我们只需要 Global 中很简单的几行代码,就可以完成整个系统的依赖注入匹配:
ContainerBuilder builder = new ContainerBuilder();
builder.RegisterGeneric(typeof(Repository<,>)).As(typeof(IRepository<,>));
Type baseType = typeof(IDependency);
builder.RegisterAssemblyTypes(assemblies)
.Where(type => baseType.IsAssignableFrom(type) && !type.IsAbstract)
.AsImplementedInterfaces().InstancePerLifetimeScope();//InstancePerLifetimeScope 保证对象生命周期基于请求
IContainer container = builder.Build();
DependencyResolver.SetResolver(new AutofacDependencyResolver(container));