使用Autofac将属性注入自定义WebViewPage

时间:2013-02-18 09:50:40

标签: asp.net-mvc-3 autofac

我正在创建一个内部应用程序框架,我们组织中的其他开发团队将使用它来构建MVC应用程序。作为其中的一部分,我正在为所有视图创建菜单结构,从配置中读取并根据当前用户的权限进行修改。要创建菜单作为框架的一部分,我已经创建了一个自定义WebViewPage实现,其中包含一个自定义HTML Helper类,需要依赖ApplicationDataReader来构建菜单。

我已经阅读了各种帖子,解释说MVC需要WebViewPage有一个无参数的构造函数,所以我需要使用属性注入。我已配置Autofac MVC3 Integration,包括注册ViewRegistrationSource。麻烦的是,当访问依赖属性时,它总是为空。

这是我正在尝试进行的调用的自定义视图页面和帮助程序:

public abstract class OuBaseViewPage<TModel> : WebViewPage<TModel> where TModel : class
{
    public OuHelper<TModel> Ou { get; set; }

    public override void InitHelpers()
    {
        base.InitHelpers();
        Ou = new OuHelper<TModel>(ViewContext, this);
    }
}

public class OuHelper<TModel> where TModel : class
{
    public OuHelper(ViewContext viewContext, IViewDataContainer viewDataContainer)
        : this(viewContext, viewDataContainer, RouteTable.Routes)
    {
    }

    public OuHelper(ViewContext viewContext, IViewDataContainer viewDataContainer, RouteCollection routeCollection)
    {
        ViewContext = viewContext;
        ViewData = new ViewDataDictionary<TModel>(viewDataContainer.ViewData);
    }

    public ViewDataDictionary<TModel> ViewData { get; private set; }

    public ViewContext ViewContext { get; private set; }

    public IList<BusinessFocusArea> ReadFocusAreas()
    {
        // this is null - yes, service location, but an isolated instance of...
        return DependencyResolver.Current.GetService<IDataReader>().Read();
    }
}

问题源于InitHelpers被调用(通过Layout.Execute())BEFORE Application_Start被调用的事实,因此没有注册任何依赖项。我知道将逻辑注入到视图中并且视图应该只是愚蠢是不好的做法,但这是一个应用程序框架,它需要执行某些设置步骤,使用框架的开发人员必须无法查看。

我可以采用更好的方法吗?

此处存在类似问题:Can Autofac inject dependencies into layout view files?

1 个答案:

答案 0 :(得分:5)

  

问题源于InitHelpers被调用的事实(通过   Layout.Execute())在调用Application_Start之前

我不认为在Application_Start之前调用某些内容。我无法重现你的问题。

以下是我所做的步骤,并且完美无缺:

  1. 使用Internet模板
  2. 创建新的ASP.NET MVC 3应用程序
  3. 安装Autofac.Mvc3 NuGet
  4. 定义虚拟接口:

    public interface IDataReader
    {
    
    }
    
  5. 虚拟实现:

    public class DataReader : IDataReader
    {
    
    }
    
  6. 定义自定义助手:

    public class OuHelper<TModel> where TModel : class
    {
        private readonly IDataReader dataReader;
        public OuHelper(ViewContext viewContext, IViewDataContainer viewDataContainer, IDataReader dataReader)
            : this(viewContext, viewDataContainer, RouteTable.Routes, dataReader)
        {
        }
    
        public OuHelper(ViewContext viewContext, IViewDataContainer viewDataContainer, RouteCollection routeCollection, IDataReader dataReader)
        {
            ViewContext = viewContext;
            ViewData = new ViewDataDictionary<TModel>(viewDataContainer.ViewData);
            this.dataReader = dataReader;
        }
    
        public ViewDataDictionary<TModel> ViewData { get; private set; }
    
        public ViewContext ViewContext { get; private set; }
    
        public IDataReader DataReader
        {
            get { return this.dataReader; }
        }
    
    }
    
  7. 使用此帮助程序定义自定义WebViewPage:

    public abstract class OuBaseViewPage<TModel> : WebViewPage<TModel> where TModel : class
    {
        public OuHelper<TModel> Ou { get; set; }
    
        public override void InitHelpers()
        {
            base.InitHelpers();
            var dataReader = DependencyResolver.Current.GetService<IDataReader>();
            Ou = new OuHelper<TModel>(ViewContext, this, dataReader);
        }
    }
    
  8. 将默认视图页面替换为~/Views/web.config中的自定义视图页面:

    <pages pageBaseType="MvcApplication1.OuBaseViewPage">
    
  9. Application_Start中配置您的容器:

    protected void Application_Start()
    {
        AreaRegistration.RegisterAllAreas();
        RegisterGlobalFilters(GlobalFilters.Filters);
        RegisterRoutes(RouteTable.Routes);
    
        var builder = new ContainerBuilder();
        builder.RegisterControllers(typeof(MvcApplication).Assembly);
        builder.RegisterType<DataReader>().As<IDataReader>();
        var container = builder.Build();
        DependencyResolver.SetResolver(new AutofacDependencyResolver(container));
    }
    
  10. 现在您可以愉快地在包括_Layout在内的所有视图中使用自定义助手,而不会出现任何问题:

    @Ou.DataReader.GetType()
    
  11. 当然,在这个例子中,我刚刚将IDataReader依赖项公开为公共属性,以说明它将始终被注入并且永远不会为null。在您的特定代码中,您当然只能使用帮助程序中的私有只读字段来完成任务。