routes.MapRoute( name: "Default", url:"{controller}/{action}/{id}", defaults: new { controller ="Home", action = "Index", id = UrlParameter.Optional } );
public void Execute(RequestContextrequestContext) { var controller = (string)requestContext.RouteData.Values["controller"]; var action =(string)requestContext.RouteData.Values["action"]; requestContext.HttpContext.Response.Write( string.Format("Controller: {0}, Action: {1}", controller,action)); }
这是最简单的controller创建方式,但是通常不这样做,因为:
我们拿到的是RequestContext对象,需要手动去生成html写到Response里面返回客户端,代码将很难维护。
集成Controller类完成一个controller
一般的,每当我们新建一个controller,都是自动继承controller类,这个类主要帮我们完成两件事情:
1. 会从路由拿到requestContext对象,解析出action以及参数,并调用
2. 提供不同的result(command模式),返回给客户端
另外,我们还可以使用filter,完成不同的crosscutting的concern,后面章节会详细介绍。
由于引入了action和result的概念,使得我们的代码更容易单元测试,起码比我们自己继承IController接口,直接操作RequestContext对象容易很多。
public class DerivedController :Controller { public ActionResult Index() { ViewBag.Message = "Hellofrom the DerivedController Index method"; return View("MyView"); } }
View代码:
@{ ViewBag.Title ="MyView"; } <h2>MyView</h2> Message: @ViewBag.Message
代码很简单,就是用Viewbag传值到View显示出来,Controller返回了最常用的result,ViewResult。
1. 从context object 直接提取
2. 通过参数传进来(由基类controller完成解析)
3. 通过model binding
Request.QueryString |
从Get 请求的url中取 |
Request.Form |
从Post请求的表单取 |
Request.Cookies |
把值放在请求的cookie里带过来 |
RouteData.Route |
从路由表里取注册的路由名 |
RouteData.Values |
从路由表获取路由配置的匿名对象 |
HttpContext.Cache |
应用程序缓存 |
HttpContext.Items |
存储一些值,只在同一请求中使用(可以在httppipeline过程的不同module,handler,以及页面传递值) |
HttpContext.Session |
当前用户的session |
TempData |
存一些临时值,取出后会被自动删除 |
另外,不建议直接从RouteData.Values直接取传来的参数值:
public ActionResult ShowWeatherForecast() { string city =(string)RouteData.Values["city"]; DateTime forDate =DateTime.Parse(Request.Form["forDate"]); return View(forDate); }
建议改写为传参:
public ActionResult ShowWeatherForecast(string city, DateTime forDate) { return View(forDate); }
1. 对于引用类型,如果RouteData没有拿到参数,那么就给null了,如果要避免接收null,可以使用默认参数
2. 对于值类型,如果没有拿到参数,就会抛出异常,因此,建议值类型总提供默认参数,或者使用Nullable类型
例如:
public ActionResult Search(stringquery= "all", int page = 1) { // ...process request... return View(); }
Controller的职责:
1. 操作domain model
2. 返回一个合适的result
操作完domainmodel之后,就要给客户端返回result了,不推荐手动去实现IController接口进行Redirect或者直接Response.Write数据给客户端。前面说过了:
1. 直接输出html,或者直接跳转url,降低了可读性和可维护性
2. 很难单元测试
3. 交接很难上手
MVC Framework 的controller已经给我们了足够的result类型来返回给客户端完成交互。为了了解ActionResult,先customize一个:
public class CustomRedirectResult: ActionResult { public string Url { get; set; } public override void ExecuteResult(ControllerContextcontext) { string fullUrl =UrlHelper.GenerateContentUrl(Url, context.HttpContext); context.HttpContext.Response.Redirect(fullUrl); } }
要customize一个result,需要继承ActionResult,主要实现ExecuteResult方法,MVCFramework给我们了一个controllerContext对象,里面有足够我们需要的信息,以上创建的actionResult功能:指定一个url,拿到controllerContext对象生成一个fullurl,完成跳转。
使用这个result:
public ActionResult ProduceOutput() { if (Server.MachineName == "IORI"){ return new CustomRedirectResult {Url = "/Basic/Index" }; } else { Response.Write("Controller:Derived, Action: ProduceOutput"); return null; } }
ViewResult |
返回一个view,可以指定不同的controller |
ParcialViewResult |
返回部分view |
RedirectToActionResult |
转到另一个action |
RedirectResult |
返回301或者302(permanent) |
JsonResult |
返回js最喜欢的json,常用,尤其当前段打算采用纯js template,或者single page application |
FileResult |
文件result |
ContentResult |
字符串 |
HttpUnauthorizedResult |
401,通常,会自动跳转到登陆页面 |
HttpNotFoundResult |
404 |
1. /Views/<ControllerName>/<ViewName>.aspx
2. /Views/<ControllerName>/<ViewName>.ascx
3. /Views/Shared/<ViewName>.aspx
4. /Views/Shared/<ViewName>.ascx
5. /Views/<ControllerName>/<ViewName>.cshtml
6. /Views/<ControllerName>/<ViewName>.vbhtml
7. /Views/Shared/<ViewName>.cshtml
8. /Views/Shared/<ViewName>.vbhtml
可以看到,mvcframework会先从熟悉的aspx和ascx找起
public ViewResult Index() { return View("~/Views/Other/Index.cshtml"); }
如果出现类似以上的代码,请思考两个问题:
想要跳转到另一个action?可以考虑使用RedirectToAction
View是否放错了位置?
public ViewResult Index() { DateTime date = DateTime.Now; return View(date); }
View中:
@model DateTime @{ ViewBag.Title ="Index"; } <h2>Index</h2> The day is: @Model.DayOfWeek
public ViewResult Index() { ViewBag.Message ="Hello"; ViewBag.Date = DateTime.Now; return View(); }
View中:
@{ ViewBag.Title ="Index"; } <h2>Index</h2> The day is:@ViewBag.Date.DayOfWeek <p /> The message is: @ViewBag.Message
ViewBag优点:
可以不需要定义类型直接传递,并且可以传递任意个对象
缺点:
错误总是运行时,由于是DynamicObject(实际是DynamicViewDataDictionary继承自DynamicObject,因此是像其他functionlanguage语言一样可以动态扩展的)
public RedirectResult Redirect(){ return Redirect("/Example/Index"); }
public RedirectResult Redirect(){ return RedirectPermanent("/Example/Index"); }
public RedirectToRouteResult Redirect() { return RedirectToRoute(new { controller = "Example", action = "Index", ID = "MyID" }); }
大多数情况,遇到处理不掉的请求,我们起码是可以确定跳转到哪个controller和action的:
public RedirectToRouteResult RedirectToRoute() { return RedirectToAction("Index"); }
可以指定controller:
public RedirectToRouteResult Redirect() { return RedirectToAction("Index", "Basic"); }
在MVCframework中,理想选择应该是使用TempData:
赋值:
public RedirectToRouteResult RedirectToRoute() { TempData["Message"] ="Hello"; TempData["Date"] =DateTime.Now; return RedirectToAction("Index"); }
取值:
public ViewResult Index() { ViewBag.Message =TempData["Message"]; ViewBag.Date =TempData["Date"]; return View(); }
这个对象的好处是,取完就被标记为removable了,请求完毕会自动清掉。如果想取多次,那么可以使用peek方法(但是还是建议最后一次取了清掉(用索引取)):
TempData.Peek("Date");
另外,笔者还推荐HttpRequest.Items,也是在httppipeline内传值的一个不错的选择。
404 (url 找不到):
public HttpStatusCodeResult StatusCode() { return new HttpStatusCodeResult(404, "URL cannot be serviced"); }
401(访问权限受限):
public HttpStatusCodeResult StatusCode() { return new HttpUnauthorizedResult(); }