在Controller操作方法中重用代码的最佳方法

joh*_* Gu 5 asp.net-mvc asp.net-mvc-3 asp.net-mvc-4

我在相同的控制器类中有一些看起来几乎相同的代码,例如设置viewbags来填充所有下拉列表,相同的代码适用于我的Post并获得Create和Edit操作方法.所以我在我的控制器类的末尾创建了一个私有方法如下: -

private void populateViewBags()
{
    string controllerName = RouteData.Values["controller"].ToString();
    ViewBag.PossibleDataCenters = repository.AllDataCenter().OrderBy(a => a.Name).ToList();
    ViewBag.PossibleZones = repository.AllZone().OrderBy(a => a.Name).ToList();
    List<string> s = new List<string>();
    s.Add(controllerName.ToLower());
    ViewBag.Products = repository.GetProducts(s).OrderBy(a => a.COMPONENTNAME).ToList();
    ViewBag.Sites = repository.GetSDOrg().OrderBy(a => a.NAME).ToList();
    ViewBag.Customers = repository.FindAccountDefinition(null).ToList();
}
Run Code Online (Sandbox Code Playgroud)

我在我的action方法中调用了这个方法.那么重用代码的正确方法是什么?谢谢

Nen*_*nad 4

有两种标准方法可以做到这一点。

第一种方法- 重写OnActionExecuting和/或OnActionExecuted控制器类的方法:

public class HomeController: Controller
{
    protected override void OnActionExecuting(ActionExecutingContext filterContext)
    {
        string controllerName = RouteData.Values["controller"].ToString();
        ViewBag.ControllerName = controllerName;
    }

    protected override void OnActionExecuted(ActionExecutedContext filterContext)
    {
        string controllerName = RouteData.Values["controller"].ToString();
        ViewBag.ControllerName = controllerName;
    }
}
Run Code Online (Sandbox Code Playgroud)

您还可以创建实现这些方法的抽象基本控制器,然后从抽象控制器继承具体控制器,这样就不会在每个控制器中重复代码。

第二种方法- 创建自定义ActionFilter属性并装饰每个需要执行其他操作的控制器。

public class MyActionFilterAttribute: ActionFilterAttribute
{
    public override void OnActionExecuting(ActionExecutingContext filterContext)
    {
        string controllerName = filterContext.RouteData.Values["controller"].ToString();
        filterContext.Controller.ViewBag.ControllerName = controllerName;
    }

    public override void OnActionExecuted(ActionExecutedContext filterContext)
    {
        string controllerName = filterContext.RouteData.Values["controller"].ToString();
        filterContext.Controller.ViewBag.ControllerName = controllerName;
    }
}
Run Code Online (Sandbox Code Playgroud)

然后只需装饰控制器,例如:

[MyActionFilter]
public class HomeController: Controller
{
    // ....
}
Run Code Online (Sandbox Code Playgroud)

更新:过滤器方法的额外灵活性,如果您需要基于每个操作进行过滤,而不是控制器中的所有操作,也可以:

public class HomeController: Controller
{
    [MyActionFilter]
    public ActionResult MyAction()
    {
        //...
    }
}
Run Code Online (Sandbox Code Playgroud)