ASP.NET MVC 3 依赖注入 - 控制器、视图和操作过滤器

2024-02-12

我正在尝试使用 Microsoft Unity 在 ASP.NET MVC 3 应用程序中进行依赖项注入。首先,我实现了自己的 IDependencyResolver 并在 Global.asax 文件中激活它,如下所示:

DependencyResolver.SetResolver(new UnityDependencyResolver(container));

我发现我不需要做任何其他事情来让控制器注入(通过构造函数和 [Dependency] 属性)工作。使用默认视图引擎,我还发现我可以让 [Dependency] 属性在标准视图中工作,但不能在布局视图中工作。是否也可以使其适用于布局视图?

不过,我已经实现了自己的视图引擎,它继承自 VirtualPathProviderViewEngine,它重写 CreateView/CreatePartialView 方法并返回我自己的自定义视图(实现 IView)。请参阅下面的自定义视图的 Render 方法:

public void Render(ViewContext viewContext, TextWriter writer) {
    var webViewPage = DependencyResolver.Current.GetService(_type) as WebViewPage;
    //var webViewPage = Activator.CreateInstance(_type) as WebViewPage;

    if (webViewPage == null)
        throw new InvalidOperationException("Invalid view type");

    webViewPage.VirtualPath = _virtualPath;
    webViewPage.ViewContext = viewContext;
    webViewPage.ViewData = viewContext.ViewData;
    webViewPage.InitHelpers();

    WebPageRenderingBase startPage = null;

    if (_runViewStartPages)
        startPage = StartPage.GetStartPage(webViewPage, "_ViewStart", _viewStartFileExtensions);

    var pageContext = new WebPageContext(viewContext.HttpContext, webViewPage, null);
    webViewPage.ExecutePageHierarchy(pageContext, writer, startPage);
}

通过注释掉的行,我完全失去了视图中的依赖注入,因此我将其更改为上面的行,该行再次适用于标准视图,但不适用于布局视图。如果您能向我展示如何修改上述内容以适用于布局视图,我将不胜感激?

最后,我试图让动作过滤器注入也正常工作。我发现了两种不同的情况:

  1. 通过属性将过滤器应用到操作。

  2. 将其定义为全局过滤器,例如:

    GlobalFilters.Filters.Add(new TestAttribute());

两者似乎都没有使用依赖解析器。因此我需要做一些额外的工作。如果有更好的方法请指正。为了启用第一个场景,我执行了以下操作:

public class UnityFilterAttributeFilterProvider : FilterAttributeFilterProvider {
    private IUnityContainer _container;

    protected override IEnumerable<FilterAttribute> GetControllerAttributes(ControllerContext controllerContext, ActionDescriptor actionDescriptor) {
        var attributes = base.GetControllerAttributes(controllerContext, actionDescriptor);

        foreach (var attribute in attributes) {
            _container.BuildUp(attribute.GetType(), attribute);
        }

        return attributes;
    }

    protected override IEnumerable<FilterAttribute> GetActionAttributes(ControllerContext controllerContext, ActionDescriptor actionDescriptor) {
        var attributes = base.GetActionAttributes(controllerContext, actionDescriptor);

        foreach (var attribute in attributes) {
            _container.BuildUp(attribute.GetType(), attribute);
        }

        return attributes;
    }
}

然后在我的 Global.asax 文件中定义它,如下所示:

FilterProviders.Providers.Remove(FilterProviders.Providers.Single(f => f is FilterAttributeFilterProvider));
FilterProviders.Providers.Add(new UnityFilterAttributeFilterProvider(container));

这工作正常,但我想知道这是否是正确的方法?为了解决第二种情况,我只需将定义全局过滤器的位置更改为以下内容:

GlobalFilters.Filters.Add(DependencyResolver.Current.GetService<TestAttribute>());

现在这又可以了,但是这是正确的方法吗?

我将不胜感激的帮助。谢谢


自从我最初问这个问题以来已经有一段时间了,但我想我会分享我最终做了什么。

在我无法使用构造函数或属性注入的情况下,我通过使用DependencyResolver(服务定位器模式)。例如,如果我需要该服务IService我会简单地像这样注入它:

public IService Service => DependencyResolver.Current.GetService<IService>();

虽然有些人可能认为这是一种反模式,但我发现它表现良好,问题更少,而且随着 C# 的新进展,我认为它看起来还不错。

但是,如果您使用 ASP.NET Core,则永远不必使用服务定位器模式,因为它已通过依赖注入作为其核心进行了重建。

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

ASP.NET MVC 3 依赖注入 - 控制器、视图和操作过滤器 的相关文章

随机推荐