这是我的场景。例如,假设我需要根据搜索条件返回汽车列表。我想要一个视图来显示结果,因为输出是相同的,但我需要几种方法来到达那里。例如,我可能有一个带有按年份搜索的文本框的表单。我可能有另一个单独的页面,其中包含所有红色丰田汽车的超链接。如何在同一个视图和控制器中处理这些多个场景。我的困境是搜索可能包含几个选项……年份、品牌、型号等,但我不知道把它们放在哪里。
最好的方法是什么?我应该在路由中定义参数还是使用查询字符串等?
这是我的场景。例如,假设我需要根据搜索条件返回汽车列表。我想要一个视图来显示结果,因为输出是相同的,但我需要几种方法来到达那里。例如,我可能有一个带有按年份搜索的文本框的表单。我可能有另一个单独的页面,其中包含所有红色丰田汽车的超链接。如何在同一个视图和控制器中处理这些多个场景。我的困境是搜索可能包含几个选项……年份、品牌、型号等,但我不知道把它们放在哪里。
最好的方法是什么?我应该在路由中定义参数还是使用查询字符串等?
选项1
当然,您始终可以选择 /car/search/?vendor=Toyota&color=Red&model=Corola 的方式,我认为这对您有好处。
routes.MapRoute(
"CarSearch",
"car/search",
new { controller = "car", action = "search" }
);
在这种情况下,您可以从 Request.Params 获取参数。
选项 2
或者您可以在路由表中定义参数,但是 AFAIK 需要为所有可能的组合制定一组规则,因为参数的顺序很重要,例如:
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,所以您需要制作自定义链接生成器。
所以,如果你需要一个非常灵活的路由——你最好用这种艰难的方式:)
控制器上的每个方法(操作)都会采用不同的参数,但会创建相同的搜索结果集合。那么,每个人都会
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));
}
}