代码改变世界

分析Autofac如何实现Controller的Ioc(Inversion of Control)

2012-10-19 21:33  JustRun  阅读(7344)  评论(13编辑  收藏  举报

Autofac是一个Ioc框架,最大的特点应该是可以不用配置文件,直接用C#代码来注册。

Autofac同时也提供了对于Asp.net MVC的扩展。

这里是Autofac的文档,介绍如何在MVC项目中集成使用:

protected void Application_Start()
{
    var builder = new ContainerBuilder();
    builder.RegisterControllers(typeof(MvcApplication).Assembly);
    var container = builder.Build();
    DependencyResolver.SetResolver(new AutofacDependencyResolver(container));

    // Other MVC setup...

 

解释一下里面的过程:

1. 首先创建一个ContainerBuilder(它会稍后为我们提供一个容器,我们可以从中取出我们所需的对象实例)

2. 注册当前Assembly中的所有Controllers到Builder,这样Builder就获取了当前MVC项目中的所有Controller类型

3. 创建容器

4. 用AutofacDependencyResolver替换MVC默认的DependencyResolver

 

OK. 到这里, 前面的还容易理解,最后一个DependencyResolver做了什么? 要了解DependencyResolver, 需要先知道ControllerFactory

MVC中的ControllerFactory

MVC请求的过程,就是根据请求的URL找到匹配的Route, Route解析出来对应的Controller的名字, 然后根据名字,找到对应的Controller类型,然后实例化一个Controller的对象响应请求。

 

上面粗体标出的,就是DefaultControllerFactory做的事情,也就是它的方法CreateController方法:

public virtual IController CreateController(RequestContext requestContext, string controllerName)

 

CreateController方法又主要依赖于GetControllerType和GetControllerInstance方法:

protected internal virtual Type GetControllerType(RequestContext requestContext, string controllerName)

protected internal virtual IController GetControllerInstance(RequestContext requestContext, Type controllerType)

 

看到这么多的virtual方法,是不是很激动? 这个不是明显是让我们继承和override的嘛。假如我们override这里的GetControllerInstance方法,然后根据这里的controllerType从Ioc容器中获取这个ControllerType的实例,不就万事OK了吗?

 

没错,继承和重写GetControllerInstance方法的确可以实现Controller的Ioc, 然后在Application_Start()里面,使用下面这行代码,替换掉DefaultControllerFactory.

ControllerBuilder.Current.SetControllerFactory(new MyControllerFactory(_container));

 

说好的DependencyResolver呢?

其实DefaultControllerFactory中的GetControllerInstance,调用了IDependencyResolver接口定义的方法GetService获取实例。通过继承这个接口,替换原有的DependencyResolver, 对于MVC原有的改动更小。所以使用通过继承IDependencyResolver 接口的方式来实现更好. 这就是开篇我们看到的:


DependencyResolver.SetResolver(new AutofacDependencyResolver(container));

IDependencyResolver借口
public interface IDependencyResolver {

       object GetService(Type serviceType);

       IEnumerableGetServices(Type serviceType);

}

 

想看看AutofacDependencyResolver的源代码是如何根据Type从容器中获取实例的, 可以到这里

 

思考和实践: 什么时候用到ControllerFactory

Asp.net MVC中的Area的使用,应该大家比较熟悉了。它是用来解决大的项目,多人开发的情况的。area实现的原理,是通过命名空间来区分即使名字相同的Controller的。可是无论如何,area都是在一个project里面的.

 

如何能够分离area到不同的project中呢? 这里的GetControllerType方法就是很好的突破口。下次探讨一下如何实现area分离到不同的project中,方便较大的项目。这也是Orchard CMS模块实现的原理。