也说Autofac在MVC的简单实践:破解在Controller构造函数中的实例化

相信大家对Autofac并不陌生,很多人都在使用。本文只是介绍一下本人在使用时的一点想法总结。


你是不是很头疼的要在Global中写一堆代码来维护Autofac?
你是不是很头疼为Controller增加构造函数为变量赋值?
你是不是很头疼每次增加接口和实现的时候都要重新编译?

那么本文介绍一些Autofac的其它实践,也许能够对你有所帮助。

在使用一个框架时,肯定要去它的官网查阅一下。autofac的官网给出了一些经典的使用案例。如注册容器:

var builder = new ContainerBuilder();
 
// Register individual components
builder.RegisterInstance(new TaskRepository)
       .As<ITaskRepository>();
builder.RegisterType<TaskController>();
builder.Register(c => new LogManager(DateTime.Now))
       .As<ILogger>();
 
// Scan an assembly for components
builder.RegisterAssemblyTypes(myAssembly)
       .Where(t => t.Name.EndsWith("Repository"))
       .AsImplementedInterfaces();
 
var container = builder.Build();

 在mvc中使用:

public class TaskController
{
  private ITaskRepository _repository;
  private ILogger _logger;
 
  // Autofac will automatically find the registered
  // values and pass them in for you.
  public TaskController(
    ITaskRepository repository,
    ILogger logger)
  {
    this._repository = repository;
    this._logger = logger;
  }
}

在这里先重点说一下在mvc中的使用,如上代码可见,在一个请求到达时,需要对controller进行实例化,而正如autofac官网所说“If there is more than one constructor on a component type, Autofac will use the constructor with the most resolvable parameters.”,如果有多个带参构造函数,autofac默认使用参数最多的构造函数。在上面代码中,即便在一个action中,你只用了_logger,那么_repository也依旧需要被autofac解析。

而在mvc的具体应用中,我们可能会使用多重继承,如下图的结构

在这种情况下,每个controller可能会有很多构造函数,在每个请求到达时,都需要实例化相当一部分的变量。本人没有研究过这种实例化是否会影响效率,只是觉得这样对于开发来讲过于繁琐,且不利于维护,代码也并不流畅。我的想法是在action中,在需要的点去实例化。

经过一些查阅,autofac官方提供了很多库,发现其中Autofac.Mef是可以用另一种方式实现达到同样的效果。文档的介绍只有一句话“The MEF integration allows you to expose extensibility points in your Autofac applications using the Managed Extensibility Framework.”  mef可能主要用来在对已经开发完毕的版本做补充的时候使用。如某个系统已经开发结束并部署运行了,这时候会有些功能的增加和扩展,在不修改原版本的前提下,使用mef可以将后补充的功能ioc到原系统。mef需要引用System.ComponentModel.Composition.dll库。

先不说别的了,代码说明一切。在接口实现上需要加入ExportAttribute,如:

    [Export(typeof(ICustomerBusiService))]
    public class CustomerBusiService : ICustomerBusiService

 注意,ICustomerBusiService不用做任何的描述,只描述其实现CustomerBusiService即可。为了达到我的目的,我在顶层的controller中增加了一个获取实例的方法,以便action中根据自己的需要获取实例化:

    public abstract class AbstractController : Controller
    {
        private static IAutofacResolver _resolver = new AutofacResolver();
        protected T GetService<T>()
        {
            return _resolver.GetService<T>();
        }
    }
下面展示一下IAutofacResolver及其实现AutofacResolver
    public interface IAutofacResolver
    {
        T GetService<T>();
    }

    public class AutofacResolver : IAutofacResolver
    {
        private Autofac.IContainer _container;

        public T GetService<T>()
        {
            if (_container == null || !_container.IsRegistered<T>())
            {
                RegisterPartsFromReferencedAssemblies();
            }
            return _container.Resolve<T>();
        }

        private void RegisterPartsFromReferencedAssemblies()
        {
            var asses = BuildManager.GetReferencedAssemblies().Cast<Assembly>();
            var assemblyCatalogs = asses.Select(x => new AssemblyCatalog(x));
            var catalog = new AggregateCatalog(assemblyCatalogs);

            var builder = new ContainerBuilder();
            builder.RegisterComposablePartCatalog(catalog);
            _container = builder.Build();
            DependencyResolver.SetResolver(new AutofacDependencyResolver(_container));
        }
    }

我们知道Asp.Net的项目,只要bin目录有变化,该站点会相应的重新启动,所以在你制作的新的接口库和实现库完成之后,只要push到bin目录,RegisterPartsFromReferencedAssemblies()中的BuildManager.GetReferencedAssemblies()会捕捉到所有的当前站点的所有dll引用,并根据ExportAttribute加载到Autofac。该方式在第一次使用GetService<T>()的时候会执行RegisterPartsFromReferencedAssemblies(),之后就不会在执行该方法了。该方式不需要在Global中用一堆代码维护Autofac,有点一劳永逸的感觉。

有了这样一个结构,那么在具体的controller中我不需要有构造函数,在acton中只要调用GetService<T>()就可以获取我需要的实例。

    public class AccountController : AbstractMvcController
    {
        [HttpPost]
        public ActionResult Register(Customer customer)
        {
            var ibsCusomter = GetService<ICustomerBusiService>();
            ibsCusomter.Register(customer);
            return View();
        }
    }

以上就是全部内容。本文并没有针对复杂的autofac应用进行说明,比如注册复杂的模型,激活事件等。只是对比较简单普遍的autofac的使用进行一些分析,个人认为mvc的站点开发中,不太会用到比较复杂的东西,因为每个用户请求都是独立的,又有并发的问题,所以autofac的单实例也基本不会考虑。

autofac相关库的下载,请点击此处

 

posted on 2013-11-14 10:10  winhu  阅读(4597)  评论(3编辑  收藏  举报

导航