我有一个动作过滤器,负责将一些常见的信息放在ViewBag中,供共享的_Layout.cshtml文件中的所有视图使用。
public class ProductInfoFilterAttribute : ActionFilterAttribute { public override void OnActionExecuting(ActionExecutingContext filterContext) { // build product info // ... (code omitted) dynamic viewBag = filterContext.Controller.ViewBag; viewBag.ProductInfo = info; } }
在共享的_Layout.cshtml文件中,我使用已经放入ViewBag的信息。
... @ViewBag.ProductInfo.Name ...
如果在处理控制器操作时发生异常,则标准HandleErrorAttribute应该显示我的共享的Error.cshtml视图,这在我介绍了上面的操作过滤器之前已经开始,并且开始使用_Layout.cshtml中的ViewBag中的新值。现在我得到的是标准的ASP.Net运行时错误页面,而不是我自定义的Error.cshtml视图。
我已经跟踪了这一点,在渲染错误视图时,在_Layout.cshtml中使用ViewBag.ProductInfo.Name引发了一个RuntimeBinderException(“在空引用上无法执行运行时绑定”)。
看来即使我的动作过滤器在抛出原始异常之前成功设置了ViewBag中的值,当渲染我的Error.cshtml视图时,会使用一个带有空的ViewBag的新上下文。
解决方法
我通过添加另一个过滤器来提出自己的解决方案。
public class PreserveViewDataOnExceptionFilter : IExceptionFilter { public void OnException(ExceptionContext filterContext) { // copy view data contents from controller to result view ViewResult viewResult = filterContext.Result as ViewResult; if ( viewResult != null ) { foreach ( var value in filterContext.Controller.ViewData ) { if ( ! viewResult.ViewData.ContainsKey(value.Key) ) { viewResult.ViewData[value.Key] = value.Value; } } } } public static void Register() { FilterProviders.Providers.Add(new FilterProvider()); } private class FilterProvider : IFilterProvider { public IEnumerable<Filter> GetFilters(ControllerContext controllerContext,ActionDescriptor actionDescriptor) { // attach filter as "first" for all controllers / actions; note: exception filters run in reverse order // so this really causes the filter to be the last filter to execute yield return new Filter(new PreserveViewDataOnExceptionFilter(),FilterScope.First,null); } } }
这个过滤器需要通过调用PreserveViewDataOnExceptionFilter.Register()在Global.asax.cs Application_Start()方法中全局连接。
我在这里做的是设置一个新的异常过滤器,最后运行HandleErrorAttribute过滤器,并将可用的ViewData集合的内容复制到控制器中,该控件将异常抛出到由HandleErrorAttribute过滤器创建的结果中。