用于search的ASP.NET MVC自定义路由

这是我的情况。 对于这个例子可以说,我需要返回一个基于search条件的汽车列表。 我想有一个视图来显示结果,因为输出将是相同的,但我需要几种方法到达那里。 例如,我可能会有一个带有文本框的表单按年进行search。 我可能有另一个单独的页面,其中包含所有红色丰田汽车的超链接。 如何在同一视图和控制器中处理这些多个场景。 我的困境是,search可能包含几个选项…年,制作,模型等,但我不知道把它们放在哪里。

什么是最好的办法呢? 我应该在路由中定义参数还是使用查询string等?

选项1

当然,你总是可以select/ car / search /?vendor = Toyota&color = Red&model = Corola的方式,我认为这对你有好处。

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

在这种情况下,您可以从Request.Params中获取params。

选项2

或者你可以在路由表中定义params,但是AFAIK需要为所有可能的组合制定一套规则,因为params的顺序很重要,例如:

  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实施scheme – 它将为您提供更多的params顺序灵活性。 但是,如果你有一个简单的应用程序,它是一个很难的方式,不要去用它。 所以,举个例子来说明这个方法(非常简单的例子):

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

 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(); } 

它将使用任何search参数顺序:

 /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 / Toyotaforms的url,所以你需要制作一个自定义链接生成器。

所以,如果你需要一个非常灵活的路由 – 你最好去这个难题:)

控制器上的每个方法(操作)都会采用不同的参数,但会创build相同的search结果集合。 那么,每个人都会

 return View("SearchResult", searchResultCollection); 

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

沿着这些线应该做你以后的事情。 注意有两种不同的操作方法,但是它们都返回到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)); } }