zoukankan      html  css  js  c++  java
  • 也说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相关库的下载,请点击此处

  • 相关阅读:
    java基础(7)--方法
    java基础(6)--数组和方法
    java基础(5)--流程控制结构
    java基础(4)--运算符及表达式
    java基础(2)--进制
    Java基础(1)--JDK,变量
    quartz(8)--其他
    spring AOP 概述(三) Advisor
    spring AOP 概述(二) Pointcut
    spring AOP 概述(一) Advice
  • 原文地址:https://www.cnblogs.com/winhu/p/Autofac_MVC.html
Copyright © 2011-2022 走看看