1、注册地址重写
MVC的地址重写必须在Global.asax.cs中初始化,从而保证所有的请求都能被Controller控制。简单的意思就是必须在运行期,应用程序启动的时候被初始化。为什么要到运行期,而不发生在编译器,为什么不能用静态构造函数来完成呢?那是因为编译期连地址都没确定,怎么能够保证地址是正确的呢?
在Global.asax.cs中默认有两个重写方法被注册
1
protected
void
Application_Start(
object
sender,
EventArgs
e)
2
{
3
//
Note: Change Url= to Url="[controller].mvc/[action]/[id]" to enable
4
//
automatic support on IIS6
5
6
RouteTable.Routes.Add(
new
Route
7
{
8
Url
=
"
[controller]/[action]/[id]
",
9
Defaults
=
new
{
action
=
"
Index
",
id
=
(
string)
null
}
,
10
RouteHandler
=
typeof(MvcRouteHandler)
11
});
12
13
14
RouteTable.Routes.Add(
new
Route
15
{
16
Url
=
"
Default.aspx
",
17
Defaults
=
new
{
controller
=
"
Home
",
action
=
"
Index
",
id
=
(
string)
null
},
18
RouteHandler
=
typeof(MvcRouteHandler)
19
});
20
21
22
}
第一个注册方法是描述了一组规则,而第二组方法描述了根目录下的
Default.aspx如何交由controller 来处理。
2、controller 处理地址
比如,我们现在的URL地址是http://www.yurow.cn/,那么,当请求http://www.yurow.cn/的时候,实际上是请求的Default.aspx,这个时候,程序找到规则Defaults = new { controller = "Home", action = "Index", id = (string)null },(这个是默认规则,所有的默认规则都可以通过Defaults = object的方式来调用,当然也有例外,下面会讲到。)。默认规则告诉程序,这个页面请求要交给一个类名为HomeController的控制器来处理(controller = "Home", 就是用控制器的类名减去Controller,这点和Attribute非常相似)。而action = "Index"告诉程序,请求的是控制器中的Index方法,最后的 id = (string)null 则是默认参数。在MVC中,默认参数只有一个,就是id。而这里id值是null。在默认HomeController类中也有两个方法
1 [ControllerAction]
2 public void Index()
3 {
4 RenderView("Index");
5 }
6
7 [ControllerAction]
8 public void About()
9 {
10 RenderView("About");
11 }
请求 Default.aspx调用Index方法后,将输出定位到Index.aspx(RenderView("Index");)。这时候显示的就是Index.aspx。当然,这时候浏览器显示的地址是http://www.yurow.cn/,实际上还可以有另外一种表示方式http://www.yurow.cn/Home/Index。
而“域名/controller/action/id” 才是MVC真正的地址现实方式。映射的地址则是“根路径/Views/controller/action(.aspx)”。id参数不是交由aspx文件处理的,而是交由controller处理,这点就是和WebForm模型最大的不同。
再来看看注册的第一组规则: Url = "[controller]/[action]/[id]",定义了URL的路径就是上面所说的“域名/controller/action/id”,在这里就是http://www.yurow.cn/Home/About 。
3、controller 到View的数据传递
这两组定义中似乎都没有用到id参数,id到底体现在哪里呢?
尝试在HomeController和About.aspx.cs中接收id参数Request["id"],然后把地址变成http://www.yurow.cn/Home/About/1,结果Request["id"]是null并不是我们想象中的'1'。而事实上是这样接收这个id参数的:
[ControllerAction]
public
void
About(
string
id)
{
if
(
string.IsNullOrEmpty(id))
id
=
"";
RenderView(
"
About
",
(
object)id);
}
在About.aspx.cs中要这样使用
(object)id,并且申明这是一个string类型。而ViewData就是这个值。当然也可以定义成任何类型,因为这里接收的类型是object。
4、重写后的地址构造
MVC提供了自动构造重写地址的方法——Html.ActionLink。
Html.ActionLink一共有3个重载。
1)先说简单的public string ActionLink(string linkText, string actionName);
参数一是链接显示的文字,而参数二是action的名称。比如Html.ActionLink("链接文字","About");
链接地址实际是<a href="/Home/About">链接文字</a>。地址中的Home是怎么来的呢?就是Defaults里定义的。注意,如果你不去改写,那么赋值了一次,无论是controller 还是action或者id的值都是一直不变的。而这个Home在注册的时候就已经被赋值了。
2)public string ActionLink(string linkText, string actionName, string controllerName);
参数一和参数二和1)相同,参数三则是controller的名称。比如你可以赋值
Html.ActionLink("链接文字","MyPage","Test");
链接地址实际是<a href="/Test/MyPage">链接文字</a>。明明没有Test控制器,也没有MyPage页面,怎么能够定义成功?这就是因为,虽然参数名是这样定义的,实际上这仅仅是一个字符串,代表要请求的目录(controllerName),和aspx文件(actionName)。
3)public string ActionLink(string linkText, object values);
这个是自定义式的赋值。可以这样写:
<%= Html.ActionLink("About Us", new { id="1",controller="Home", action="About" })%>
MVC的地址重写必须在Global.asax.cs中初始化,从而保证所有的请求都能被Controller控制。简单的意思就是必须在运行期,应用程序启动的时候被初始化。为什么要到运行期,而不发生在编译器,为什么不能用静态构造函数来完成呢?那是因为编译期连地址都没确定,怎么能够保证地址是正确的呢?
在Global.asax.cs中默认有两个重写方法被注册
10
11
12
13
14
15
16
17
18
19
20
21
22
2、controller
比如,我们现在的URL地址是http://www.yurow.cn/,那么,当请求http://www.yurow.cn/的时候,实际上是请求的Default.aspx,这个时候,程序找到规则Defaults
10
11
请求 Default.aspx调用Index方法后,将输出定位到Index.aspx(RenderView("Index");)。这时候显示的就是Index.aspx。当然,这时候浏览器显示的地址是http://www.yurow.cn/,实际上还可以有另外一种表示方式http://www.yurow.cn/Home/Index。
而“域名/controller/action/id”
再来看看注册的第一组规则:
3、controller
这两组定义中似乎都没有用到id参数,id到底体现在哪里呢?
尝试在HomeController和About.aspx.cs中接收id参数Request["id"],然后把地址变成http://www.yurow.cn/Home/About/1,结果Request["id"]是null并不是我们想象中的'1'。而事实上是这样接收这个id参数的:
1
public
partial
class
About
:
ViewPage<
string>
2
{
3
4
protected
void
Page_Load(
object
sender,
EventArgs
e)
5
{
6
Response.Write(ViewData);
7
}
8
}
ViewPage<string>定义了About.aspx.cs接收Controller里传递过来的 2
3
4
5
6
7
8
4、重写后的地址构造
MVC提供了自动构造重写地址的方法——Html.ActionLink。
Html.ActionLink一共有3个重载。
1)先说简单的public string ActionLink(string linkText, string actionName);
参数一是链接显示的文字,而参数二是action的名称。比如Html.ActionLink("链接文字","About");
链接地址实际是<a href="/Home/About">链接文字</a>。地址中的Home是怎么来的呢?就是Defaults里定义的。注意,如果你不去改写,那么赋值了一次,无论是controller
2)public string ActionLink(string linkText, string actionName, string controllerName);
参数一和参数二和1)相同,参数三则是controller的名称。比如你可以赋值
Html.ActionLink("链接文字","MyPage","Test");
链接地址实际是<a href="/Test/MyPage">链接文字</a>。明明没有Test控制器,也没有MyPage页面,怎么能够定义成功?这就是因为,虽然参数名是这样定义的,实际上这仅仅是一个字符串,代表要请求的目录(controllerName),和aspx文件(actionName)。
3)public string ActionLink(string linkText, object values);
这个是自定义式的赋值。可以这样写:
<%= Html.ActionLink("About Us", new { id="1",controller="Home", action="About" })%>
实际地址就是<a href="/Home/About/1">About Us"</a>。