ASP.NET MVC自定义路由用于搜索

31

这是我的情境。假设我需要根据搜索条件返回一辆汽车列表作为示例。由于输出相同,我希望有一个单独的视图来显示结果,但是我需要多种方式来实现搜索。例如,我可能有一个包含文本框以按年份搜索的表单。我可能会有另一个单独页面,其中包含所有红色丰田车的超链接。如何在同一个视图和控制器中处理这些多个方案?我的困境在于搜索可能包含多个选项……年份、品牌、型号等,但我不知道应该放在哪里。

对此,最佳方法是什么?应该在路由中定义参数还是使用查询字符串等等?

3个回答

78

选项1

当然你可以一直选择这样的方式/car/search/?vendor=Toyota&color=Red&model=Corola,我认为这对你来说是不错的选择。

routes.MapRoute(
    "CarSearch",
    "car/search",
    new { controller = "car", action = "search" }
);

在这种情况下,您可以从Request.Params中获取操作参数。

选项2

或者,您可以在路由表中定义参数,但据我所知,将需要为所有可能的组合制定一套规则,因为参数的顺序很重要,例如:

        routes.MapRoute(
            "CarSearch1",
            "car/search/vendor/{vendor}/color/{color}/model/{model}",
            new {controller = "car", action = "search"}
        );

        routes.MapRoute(
            "CarSearch2",
            "car/search/color/{color}/vendor/{vendor}/model/{model}",
            new {controller = "car", action = "search"}
        );

        routes.MapRoute(
            "CarSearch3",
            "car/search/model/{model}/color/{color}/vendor/{vendor}",
            new {controller = "car", action = "search"}
        );

如果您使用标准的MvcRouteHandler,则这是正确的。

但是,有一个简单的方法 :)

选项3

最困难但我认为最优雅的方法是创建自己的IRouteHandler实现——它将在参数顺序方面给您更大的灵活性。但同样需要指出,如果您只有一个简单的应用程序,不要选择这种方式。 以下是一个制作方式的示例(非常简单):

将新路由添加到路由列表中:

routes.Add
    (
        new Route
            (
                "car/search/{*data}",
                new RouteValueDictionary(new {controller = "car", action = "search", data = ""}),
                new MyRouteHandler()
            )
    );

添加类,以调整标准请求处理链:

class MyRouteHandler : IRouteHandler
{
    public IHttpHandler GetHttpHandler(RequestContext requestContext)
    {
        return new MyHttpHandler(requestContext);
    }
}

class MyHttpHandler : MvcHandler
{
    public MyHttpHandler(RequestContext requestContext) : base(requestContext)
    {
    }

    protected override void ProcessRequest(HttpContextBase httpContext)
    {
        IController controller = new CarController();
        (controller as Controller).ActionInvoker = new MyActionInvoker();
        controller.Execute(RequestContext);
    }
}

class MyActionInvoker : ControllerActionInvoker
{
    protected override ActionResult InvokeActionMethod(MethodInfo methodInfo, IDictionary<string, object> parameters)
    {
        // if form of model/{model}/color/{color}/vendor/{vendor}
        var data = ControllerContext.RouteData.GetRequiredString("data");
        var tokens = data.Split('/');

        var searchParams = new Dictionary<string, string>();
        for (var i = 0; i < tokens.Length; i++)
        {
            searchParams.Add(tokens[i], tokens[++i]);
        }

        parameters["searchParams"] = searchParams;

        return base.InvokeActionMethod(methodInfo, parameters);
    }
}
在控制器中:
public ActionResult Search(IDictionary<string, string> searchParams)
{
    ViewData.Add
        (
            // output 'model = Corola, color = red, vendor = Toyota'
            "SearchParams",
            string.Join(", ", searchParams.Select(pair => pair.Key + " = " + pair.Value).ToArray())
        );
    return View();
}

它将与任何搜索参数顺序一起使用:

/car/search/vendor/Toyota/color/red/model/Corola
/car/search/color/red/model/Corola/vendor/Toyota
/car/search/model/Corola/color/red/vendor/Toyota

但也别忘了编写链接生成逻辑,因为Html.ActionLink和Html.RenderLink不能以/ car / search / model / Corola / color / red / vendor / Toyota的漂亮形式提供url,所以你需要制作自定义链接生成器。

因此,如果你需要一个真正灵活的路由 - 最好走这条艰难的路:)


我认为目前最好的选择是选项1。一开始我走的是选项2的路线,但是把所有可能的路由组合起来似乎效率低下。如果我想重构,选项3看起来不错,但作为一个MVC新手,我会坚持走简单的路线。 - Papa Burgundy
我觉得我还没有足够的积分来点赞,但我会努力的 ;) - Papa Burgundy
@maxnk 你知道有没有这方面的虚拟项目? - CR41G14
@CR41G14 我不知道有这样的项目,但你可以使用这段代码作为启动来制作自己的定制解决方案。只需注意它是在MVC 1.0时编写的,因此可能需要进行一些小的调整。 - maxnk

0

控制器上的每个方法(操作)都会使用不同的参数,但创建相同的搜索结果集合。然后,每个方法都会

return View("SearchResult", searchResultCollection);

他们都使用相同的视图,SearchResult.aspx。


0

大致上这样做应该可以满足您的需求。请注意,有两种不同的操作方法,但它们都返回对DisplayResults()的调用 - 因此它们最终使用相同的视图,具有不同的ViewData。

public class SearchController : Controller {

    public ActionResult ByColor(Color[] colors) {
         List<Car> results = carRepository.FindByColor(colors);
         return(DisplayResults(result));
    }

    public ActionResult ByMake(string make) {
         List<Car> results = carRepository.FindByMake(make);
         return(DisplayResults(results));
    }

    private ActionResult DisplayResults(IList<Car> results) {

        // Here we explicitly return the view /Views/Search/Results.aspx
        // by specifying the view name in the call to View();
        return(View("Results", results));
    }
}

谢谢。你的例子是我思考的起点,但是我觉得我对MVC路由的理解不够,所以卡住了。我只看了几天。你怎么样将一个颜色数组传递给ActionResult呢?我的理解是这些参数来自URL。 - Papa Burgundy

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接