上一主题下一主题
«123»Pages: 1/3     Go
关键字
主题 : ASP.NET MVC4 URL Routing 深入与使用URL最佳实践
级别: VIP四级

UID: 230365
精华: 12
发帖: 314
威望: 3432 点
学点: 380 点
贡献: 2 点
好评: 1 点
学币: 0 个
注册时间: 2010-04-16
最后登录: 2014-09-25
楼主  发表于: 2014-07-16 16:38||

0 ASP.NET MVC4 URL Routing 深入与使用URL最佳实践

管理提醒: 本帖被 zylzyl 设置为精华(2014-07-17)
l7nc8K  
+q6/'ErN]m  
}[\l$sS  
P|l62!m<   
ASP.NET MV**** URL Routing 深入与使用URL最佳实践 ;3'ta!.c  
FxW~Co  
`$Z:j;F  
目录: i]N jn k  
前言 W9+H /T7!  
URL Routing 的定义方式 jb|al[p\  
示例准备 jJml[iC  
给片段变量定义默认值 aA7}>  
定义静态片段 hSr2<?yk  
自定义片段变量  d|;S4m`  
路由约束 Pq u]?X  
定义请求磁盘文件路由 Se{x-vn?p  
生成URL(链接) : 7>oFz  
URL方案最佳实践 |Tv}leJF  
{%Cb0Zh  
内容: ,a 2(h  
1. 前言 =S+*= jA  
我们知道在ASP.NET Web Forms中,一个URL请求往往对应一个aspx页面,一个aspx页面就是一个物理文件,它包含对请求的****。 &YpViC4K.  
而在ASP.NET MVC中,一个URL请求是由对应的一个Controller中的Action来****的,由URL Routing来告诉MVC如何定位到正确的Controller和Action。 !:Z lVIA  
总的来说,URL Routing包含两个主要功能:解析URL 和 生成URL,本文将围绕这两个大点进行讲解。 3u[m? Vw  
2. URL Routing 的定义方式 R RRF/Z;))  
让我们从下面这样一个简单的URL****始: (w[#h9j  
http://mysite.com/Admin/Index 03p D<  
在域名的后面,默认使用“/”来对URL进行分段。路由系统通过类似于 {controller}/{action} 格式的字符串可以知道这个URL的 Admin 和 Index 两个片段分别对应Controller和Action的名称。 qtrN=c3x  
默认情况下,路由格式中用“/”分隔的段数是和URL域名的后面的段数是一致的,比如,对于{controller}/{action} 格式只会匹配两个片段。如下表所示: NVnId p  
#ME!G/  
URL路由是在MVC工程中的App_Start文件夹下的RouteConfig.cs文件中的RegisterRoutes方法中定义的,下面是创建一个空MVC项目时系统生成的一个简单URL路由定义: $|"Y|3&X  
public static void RegisterRoutes(RouteCollection routes) { P K+rr.k]  
    routes.IgnoreRoute("{resource}.axd/{*pathInfo}"); `^v=*&   
    routes.MapRoute( 2NMg+Lt8v  
        name: "Default", =y^`yv 3  
        url: "{controller}/{action}/{id}", S2i*Li  
        defaults: new { controller = "Home", action = "Index",  id = UrlParameter.Optional } $-=xG&fSz  
    ); 0 i'bo*  
} Y5{KtW  
静态方法RegisterRoutes是在Global.asax.cs文件中的Application_Start方法中被调用的,除了URL路由的定义外,还包含其他的一些MVC核心特性的定义: %?`O .W  
protected void Application_Start() { s5/u>d  
    AreaRegistration.RegisterAllAreas(); .} O@<t  
    WebApiConfig.Register(GlobalConfiguration.Configuration); HKL/ D  
    FilterConfig.RegisterGlobalFilters(GlobalFilters.Filters); 'aWqj+Wbh  
    RouteConfig.RegisterRoutes(RouteTable.Routes); dAWB.#  
    BundleConfig.RegisterBundles(BundleTable.Bundles); DFkDlx  
} 29AE B  
RouteConfig.RegisterRoutes方法中传递的是 RouteTable 类的静态 Routes 属性,返回一个RouteCollection的实例。其实,“原始”的定义路由的方法可以这样写: 3D-0 N0o  
public static void RegisterRoutes(RouteCollection routes) { 7{Lp/z%r  
qRSoF04!R  
    Route myRoute = new Route("{controller}/{action}", new MvcRouteHandler()); Oq^t[X'  
    routes.Add("MyRoute", myRoute); p*j>s \  
} tQ *?L  
创建Route对象时用了一个URL格式字符串和一个MvcRouteHandler对象作为构造函数的参数。不同的ASP.NET技术有不同的RouteHandler,MVC用的是MvcRouteHandler。 bf=!\L$  
这种写法有点繁琐,一种更简单的定义方法是: w@"Zjbs`  
public static void RegisterRoutes(RouteCollection routes) { 0&u=(;Dr\  
    routes.MapRoute("MyRoute", "{controller}/{action}"); `Ez8!d{MD8  
} R; w$_1  
这种方法简洁易读,一般我们都会用这种方法定义路由。 +DG-MM%\  
3. 示例准备 2$o#b .  
作为演示,我们先来准备一个Demo。创建一个标准的MVC应用程序,然后添加三个简单的Controller,分别是HomeController、CustomerController和AdminController,代码如下: \QQWhwE  
public class HomeController : Controller { Y<IuwS  
             pIV |hb!G  
    public ActionResult Index() { G1-r$7\  
        ViewBag.Controller = "Home"; [,MaAB  
        ViewBag.Action = "Index"; EKO~\d  
        return View("ActionName"); S\]9mHJI  
    } (t){o> l  
} < )pPq+  
public class CustomerController : Controller { W\($LD"X  
         MQDLC7Y.p5  
    public ActionResult Index() { C?k\5AzT  
        ViewBag.Controller = "Customer"; .Tm.M7  
        ViewBag.Action = "Index"; P* i 'uN  
        return View("ActionName"); @{U UB=}9  
    } skXzck  
b) x0;8<  
    public ActionResult List() { c#CV5J\Kk3  
        ViewBag.Controller = "Customer"; V;g) P  
        ViewBag.Action = "List"; 7x77s  
        return View("ActionName"); ,}khu  
    } 6&5D4 V  
} ~ES%=if~Y  
public class AdminController : Controller { *zN~x(0{E  
         wx`.  
    public ActionResult Index() { ~QlF(@u e  
        ViewBag.Controller = "Admin"; rW|%eT*/'A  
        ViewBag.Action = "Index"; %~k>$(u6  
        return View("ActionName"); 6]Q#4  
    } 7.Z-  
} TQKcPVlE  
在 /Views/Shared 文件夹下再给这三个Controller添加一个共享的名为 ActionName.cshtml 的 View,代码如下: TBrw ir  
@{ F-b]>3r  
    Layout = null; XN6$TNsD$  
} XkkzY5rxOc  
<!DOCTYPE html> :*h1ik4t  
<html> U^|T{g+O  
<head> BfCM\ij  
    <meta name="viewport" content="width=device-width" /> $Zo|t a^  
    <title>ActionName</title> z/h]Jos  
</head> 4h_YVG]ur  
<body> [zm@hxym  
    <div>The controller is: @ViewBag.Controller</div> 2L;=wP2?{  
    <div>The action is: @ViewBag.Action</div> WEX7=^k9  
</body> [e ztu9  
</html> kvbW^pl  
我们把RouteConfig.cs文件中项目自动生成的URL Rounting的定义****了,然后根据前面讲的路由定义知识,我们自己写一个最简单的: ]UEA"^  
public static void RegisterRoutes(RouteCollection routes) { RQ[/s lg  
    routes.MapRoute("MyRoute", "{controller}/{action}"); vdvnwzp!l  
} )D#}/3s  
程序运行,URL定位到 Admin/Index 看看运行结果: j;E$7QH[  
%i/|}K  
这个Demo输出的是被调用的Controller和Action名称。 gJuK%P  
4 给片段变量定义默认值 q78OP}  
在上面我们必须把URL定位到特定Controller和Action,否则程序会报错,因为MVC不知道去执行哪个Action。 我们可以通过指定默认值来告诉MVC当URL没有给出对应的片段时使用某个默认的值。如下给controller和action指定默认值: IAGY-+8e  
routes.MapRoute("MyRoute", "{controller}/{action}",  new { controller = "Home", action = "Index" }); 41d+z>a]  
这时候如果在URL中不****action片段的值或不****controller和action两个片段的值,MVC将使用路由定义中****的默认值: 9O-~Ws ;  
,aUbB8  
它的各种匹配情况如下表所示: /;Hqv `X7  
QN$s %&O  
注意,对于上面的URL路由的定义,我们可以只给action一个片段指定默认值,但是不能只给controller一个片段指定默认值,即如果我们给Controller指定了默认值,就一定也要给action指定默认值,否则URL只有一个片段时,这个片段匹配给了controller,action将找不到匹配。 L[9Kh&c  
5 定义静态片段 XC~"T6F  
并不是所有的片段都是用来作为匹配变量的,比如,我们想要URL加上一个名为Public的固定前缀,那么我们可以这样定义: o938!jML_  
routes.MapRoute("", "Public/{controller}/{action}",  new { controller = "Home", action = "Index" }); +x`pWH]2  
这样,请求的URL也需要一个Public前缀与之匹配。我们也可以把静态的字符串放在大括号以外的任何位置,如: "|dhmV[;  
routes.MapRoute("", "X{controller}/{action}",  new { controller = "Home", action = "Index" }); b/m.VL  
在一些情况下这种定义非常有用。比如当你的网站某个链接已经被用户普遍记住了,但这一块功能已经有了一个新的版本,但调用的是不同名称的controller,那么你把原来的controller名称作为现在controller的别名。这样,用户依然使用他们记住的URL,而导向的却是新的controller。如下使用Shop作为Home的一个别名: 0{ B<A^Bf  
routes.MapRoute("ShopSchema", "Shop/{action}",  new { controller = "Home" }); M[z1B!rT  
这样,用户使用原来的URL可以访问新的controller: v@GhwL  
?.F^Oi6 u  
6 自定义片段变量 ;qUB[Kw  
自定义片段变量的定义和取值 9#hp]0S6  
contrlloer和action片段变量对MVC来说有着特殊的意义,在定义一个路由时,我们必须有这样一个概念:contrlloer和action的变量值要么能从URL中匹配得到,要么由默认值****,总之一个URL请求经过路由系统交给MVC****时必须保证contrlloer和action两个变量的值都有。当然,除了这两个重要的片段变量,我们也可从通过自定义片段变量来从URL中得到我们想要的其它信息。如下自定义了一个名为Id的片段变量,而且给它定义了默认值: U _A'/p^D  
routes.MapRoute("MyRoute", "{controller}/{action}/{id}", ge?or]T1S  
    new{ Tq )hAZ  
        controller = "Home", :/5G Hfyj  
        action = "Index", XgnNYy6W  
        id = "DefaultId"}); %B#T"=Cx  
}); {Y2 J: x  
我们在HomeController中增加一个名为CustomVariable的ACtion来演示一下如何取自定义的片段变量: sm{0o$\Z  
publicActionResult CustomVariable() { g| 3bM  
    ViewBag.Controller = "Home"; S},Cz  
    ViewBag.Action = "CustomVariable"; /8l@n dZf  
    ViewBag.CustomVariable = RouteData.Values["id"]; j[\:#/J  
    return View("ActionName"); H}G 9gi  
} tl~ZuS/  
可以通过 RouteData.Values[segment] 来取得任意一个片段的变量值。 z3L=K9)  
再稍稍改一下ActionName.cshtml 来看一下我们取到的自定义片段变量的值: D~i5E9s5  
... a]5y CBm  
<div>The controller is: @ViewBag.Controller</div> 1Vz^?t:  
<div>The action is: @ViewBag.Action</div> ZDp^k{AN9a  
<div>The custom variable is: @ViewBag.CustomVariable</div>... t$t'{*t( T  
将URL定位到 /Home/CustomVariable/Hello 将得到如下结果: GKF!GbGR@  
0iR?r+|  
自定义的片段变量用处很大,也很灵活,下面介绍一些常见的用法。 r 9M3rj]  
7 将自定义片段变量作为Action方法的参数 &XI9%h9|  
我们可以将自定义的片段变量当作参数传递给Action方法,如下所示: @f|~$$k=  
public ActionResult CustomVariable(string id) { L.-qTh^P  
    ViewBag.Controller = "Home"; -Q<OSa='  
    ViewBag.Action = "CustomVariable"; kxEq_FX  
    ViewBag.CustomVariable =id; "O$WfpKX  
    return View("ActionName"); XI$W   
} g n.)_  
效果和上面是一样的,只不过这样省去了用 RouteData.Values[segment] 的方式取自定义片段变量的麻烦。这个操作背后是由模型绑定来****的 [hH>BEtm  
8 指定自定义片段变量为可选 <1lB[:@%U  
指定自定片段变量为可选,即在URL中可以不用指定片段的值。如下面的定义将Id定义为可选: $39TP@?:Z)  
routes.MapRoute("MyRoute", "{controller}/{action}/{id}", new { KI*b We  
        controller = "Home", V(LfFO{^>?  
        action = "Index", e&ZH 1^O  
        id=UrlParameter.Optional Z]2z*XD  
}); @44*<!da  
定义为可选以后,需要对URL中没有Id这个片段值的情况进行****,如下: ,+OVRc  
public ActionResult CustomVariable(string id) { @Ss  W  
    ViewBag.Controller = "Home"; r+#g  
    ViewBag.Action = "CustomVariable"; G#NbLj`h  
    ViewBag.CustomVariable = id == null ? "<no value>" : id; +G!# /u1  
    return View("ActionName"); J%q)6&  
} = 4|"<8'  
当Id是整型的时候,参数的类型需要改成可空的整型(即int? id)。 ]0(ZlpT  
为了省去判断参数是否为空,我们也可以把Action方法的id参数也定义为可选,当没有****Id参数时,Id使用默认值,如下所示: n) HV:8j~  
public ActionResult CustomVariable(string id = "DefaultId") { {A , w%  
    ViewBag.Controller = "Home"; N;}X$b5Y @  
    ViewBag.Action = "CustomVariable"; ?l @=}WN  
    ViewBag.CustomVariable = id; yF |28KJ  
    return View("ActionName"); Ag(JSVY  
} SjA'<ZX>TM  
这样其实就是和使用下面这样的方式定义路由是一样的: &FZ~n?;hQ  
routes.MapRoute("MyRoute", "{controller}/{action}/{id}", new { controller = "Home", action = "Index", id = "DefaultId" }); ,*x/L?.Z!  
9 定义可变数量的自定义片段变量 &i`(y>\  
我们可以通过 catchall 片段变量加 * 号前缀来定义匹配任意数量片段的路由。如下所示: 0f3>s>`M  
routes.MapRoute("MyRoute", "{controller}/{action}/{id}/{*catchall}", CL(D&8v8~  
new { controller = "Home", action = "Index",  id = UrlParameter.Optional }); G2>s#Y5(,  
这个路由定义的匹配情况如下所示: 4/S3hH  
HG5E,^1n  
使用*catchall,将匹配的任意数量的片段,但我们需要自己通过“/”分隔catchall变量的值来取得独立的片段值。 2Mw`  
10 路由约束 { Rxb_9  
正则表达式约束 &wkb r2P  
通过正则表达式,我们可以制定限制URL的路由规则,下面的路由定义限制了controller片段的变量值必须以 H 打头: ~@}n}aV'!  
我们可以用正则表达式约束来定义只有指定的几个特定的片段值才能进行匹配,如下所示: R"F:(  
routes.MapRoute("MyRoute", "{controller}/{action}/{id}", t""Y -M  
    new { controller = "Home", action = "Index", id = UrlParameter.Optional }, :F"NF  
    new { controller = "^H.*", action = "^Index$|^About$" } )U}`x }:,  
); 9r 8bSV3`  
这个定义,限制了只能访问controller以H打着的,action片段值只能是Index或About,不区分大小写。 A*]sN8  
Http请求方式约束 { "f} }}l  
我们还可以限制路由只有当以某个特定的Http请求方式才能匹配。如下限制了只能是Get请求才能进行匹配: Gch3|e  
routes.MapRoute("MyRoute", "{controller}/{action}/{id}", Te^_gdf  
    new { controller = "Home", action = "Index", id = UrlParameter.Optional },  [cfXcl  
    new { controller = "^H.*", httpMethod = new HttpMethodConstraint("GET") } zItGoJu  
); 9wC='  
通过创建一个 HttpMethodConstraint 类的实例来定义一个Http请求方式约束,构造函数传递是允许匹配的Http方法名。这里的httpMethod属性名不****娑ǖ模皇俏饲帧 G/1V4-@  
这种约束也可以通过HttpGet或HttpPost过滤器来实现。 a5`9mR)Y$'  
自定义路由约束 Cv`dK=n>  
如果标准的路由约束满足不了你的需求,那么可以通过实现 IRouteConstraint 接口来定义自己的路由约束规则。 ,A0v 5Q<  
我们来****一个限制浏览器版本访问的路由约束。在MVC工程中添加一个文件夹,取名Infrastructure,然后添加一个 UserAgentConstraint 类文件,代码如下: 97}l`z;Z  
public class UserAgentConstraint : IRouteConstraint { ,!^;<UR:  
         R:44Gv7  
    private string requiredUserAgent; `4q tmbj  
=u5( zaBe  
    public UserAgentConstraint(string agentParam) { u6*0% Km  
        requiredUserAgent = agentParam; vH"^a/95|  
    } +3R/g@n  
% )}rQqQ  
    public bool Match(HttpContextBase httpContext, Route route, string parameterName, !WD^To  
        RouteValueDictionary values, RouteDirection routeDirection) { ^`9O$.'@  
             % hRH80W|  
        return httpContext.Request.UserAgent != null Tnnj8I1v  
            && httpContext.Request.UserAgent.Contains(requiredUserAgent); m*>gG{3;  
    } XK@Ct eP"  
} 42PA?^xPw  
这里实现IRouteConstraint的Match方法,返回的bool值告诉路由系统请求是否满足自定义的约束规则。我们的UserAgentConstraint类的构造函数接****一个浏览器名称的关键字作为参数,如果用户的浏览器包含****的关键字才可以访问。接一来,我们需要****自定的路由约束: ?T>)7Y)  
public static void RegisterRoutes(RouteCollection routes) { wQOIUvd  
xGw| @d  
    routes.MapRoute("ChromeRoute", "{*catchall}", AaDMX,  
        new { controller = "Home", action = "Index"}, AZbFj-^4  
        new { customConstraint = new UserAgentConstraint("Chrome") } 2-@z-XKn  
    ); " 8g\UR"[  
} Y]"lcr}  
下面分别是IE10和Chrome浏览器请求的结果: sbq:8P#  
)1Z @}o 9  
11 定义请求磁盘文件路由 =O~Y6|  
并不是所有的URL都是请求controller和action的。有时我们还需要请求一些资源文件,如图片、html文件和JS库等。 8 v<*xy  
我们先来看看能不能直接请求一个静态Html文件。在项目的Content文件夹下,添加一个html文件,内容随意。然后把URL定位到该文件,如下图: Ak@y"!wnM  
[img]http://bbs.ibeifeng.com/attachment/637_459674_42215b5d1cf9e27.png[/img ^-qz!ib  
我们看到,是可以直接访问一静态资源文件的。 Ev1gzHd!i  
默认情况下,路由系统先检查URL是不是请求静态文件的,如果是,**器直接返回文件内容并结束对URL的路由解析。我们可以通过设置 RouteCollection的 RouteExistingFiles 属性值为true 让路由系统对静态文件也进行路由匹配,如下所示: 8MBvp*  
public static void RegisterRoutes(RouteCollection routes) { Lz`_&&6  
     -y8> c0u  
    routes.RouteExistingFiles = true; %ERR^  
_5v]69C#  
    routes.MapRoute("MyRoute", "{controller}/{action}/{id}/{*catchall}", r<_qU3Eaj  
        new { controller = "Home", action = "Index", id = UrlParameter.Optional O+ J0X*&x  
    }); zB#.EW  
} wRe2sjM  
设置了routes.RouteExistingFiles = true后,还需要对IIS进行设置,这里我们以IIS Express为例,右键IIS Express小图标,选择“显示所有应用程序”,弹出如下窗口: 3u1\z se  
G5!!^p~  
点击并打****配置文件,Control+F找到UrlRoutingModule-4.0,将这个节点的preCondition属性改为空,如下所示: 3ZX#6*(}2  
<add name="UrlRoutingModule-4.0" type="System.Web.Routing.UrlRoutingModule" preCondition=""/> [g$IN/o%  
然后我们运行程序,再把URL定位到之前的静态文件: #++D|oE  
Poxo c-s  
这样,路由系统通过定义的路由去匹配RUL,如果路由中没有定义该静态文件的匹配,则会报上面的错误。 UMlvu?u2p1  
一旦定义了routes.RouteExistingFiles = true,我们就要为静态文件定义路由,如下所示: w,;ox2  
public static void RegisterRoutes(RouteCollection routes) { 8l>YpS*S^  
     fY 10a_@x  
    routes.RouteExistingFiles = true; nSq$,tk(  
Y|-:z@n6C  
    routes.MapRoute("DiskFile", "Content/StaticContent.html", <P-AlHYV-  
        new { controller = "Customer", action = "List", }); f:ep~5] G  
Kj#h9e  
    routes.MapRoute("MyRoute", "{controller}/{action}/{id}/{*catchall}", ``Rg0o  
        new { controller = "Home", action = "Index", id = UrlParameter.Optional }); 6\6g-1B`  
} a0hgF_O1  
这个路由匹配Content/StaticContent.html的URL请求为controller = Customer, action = List。我们来看看运行结果: A}sdi4[`  
#kAk d-QY6  
这样****的目的是为了可以在Controller的Action中控制对静态资源的请求,并且可以阻止对一些特殊资源文件的访问。 gTg[!}_;\N  
设置了RouteExistingFiles属性为true后,我们要为允许用户请求的资源文件进行路由定义,如果每种资源文件都去定义相应的路由,就会显得很繁琐。 [m9=e-KS$Q  
我们可以通过RouteCollection类的IgnoreRoute方法绕过路由定义,使得某些特定的静态文件可以由**器直接返回给给浏览器,如下所示: %TzdpQp"  
public static voidRegisterRoutes(RouteCollection routes) { bl4I4RB  
     eyG[1EEU  
    routes.RouteExistingFiles = true; B'D~ Q  
^aY,Wq  
    routes.IgnoreRoute("Content/{filename}.html"); Bx0=D:j  
^1X 6D H`  
    routes.MapRoute("DiskFile", "Content/StaticContent.html", c5 t7X-LB  
        new { controller = "Customer", action = "List", }); &M2fcw?  
?cgb3^R'  
    routes.MapRoute("MyRoute", "{controller}/{action}/{id}/{*catchall}", |7b@w;q,D  
        new { controller = "Home", action = "Index", id = UrlParameter.Optional }); gHVD,Jr  
} 8lF:70wia  
这样,只要是请求Content目录下的任何html文件都能被直接返回。这里的IgnoreRoute方法将创建一个RouteCollection的实例,这个实例的Route Handler 为 StopRoutingHandler,而不是 MvcRouteHandler。运行程序定位到Content/StaticContent.html,我们又看到了之前的静态面面了。 'PdUSv|lH  
12 生成URL(链接) iG=XRctgj)  
前面讲的都是解析URL的部分,现在我们来看看如何通过路由系统在View中生成URL。 zO3}c3D~q  
生成指向当前controller的action链接 laaoIL^  
在View中生成URL的最简单方法就是调用Html.ActionLink方法,如下面在 Views/Shared/ActionName.cshtml 中的代码所示: nLnzl  
... Z8WBOf* ~e  
<div>The controller is: @ViewBag.Controller</div> "mAMfV0  
<div>The action is: @ViewBag.Action</div> ilFS9A3P  
<div>    @Html.ActionLink("This is an outgoing URL", "CustomVariable") iayxN5,  
</div>... t&rr;W]  
这里的Html.ActionLink方法将会生成指向View对应的Controller和第二个参数指定的Action,我们可以看看运行后页面是如何显示的: #v:A-u  
q8 Rep  
经过查看Html源码,我们发现它生成了下面这样的一个html链接: pe-%`1iC0>  
<a href="/Home/CustomVariable">This is an outgoing URL</a> ?,FL"ye  
这样看起来,通过Html.ActionLink生成URL似乎并没有直接在View中自己写一个<a>标签更直接明了。 但它的好处是,它会自动根据路由配置来生成URL,比如我们要生成一个指向HomeContrller中的CustomVariable Action的连接,通过Html.ActionLink方法,只需要给出对应的Controller和Action名称就行,我们不需要关心实际的URL是如何组织的。举个例子,我们定义了下面的路由: yeN(_t2.  
public static void RegisterRoutes(RouteCollection routes) { 1Ewg_/R  
             4 zipgw  
    routes.MapRoute("NewRoute", "App/Do{action}", new { controller = "Home"}); Ms=11C  
             ~JohcU}d  
    routes.MapRoute("MyRoute", "{controller}/{action}/{id}", :6}cczQE|O  
        new { controller = "Home", action = "Index", id = UrlParameter.Optional }); v#$}3+KVC  
} UymhBh  
运行程序,我们发现它会自动生成下面这样的连接: #L xfE<^  
<a href="/App/DoCustomVariable">This is an outgoing URL</a> /5C>7BC  
所以我们要生成指向某个Action的链接时,最好使用Html.ActionLink方法,否则你很难保证你手写的连接就能定位到你想要的Action。 q%wF=<W  
生成其他controller的action链接 a,\GOy(q{  
上面我们给Html.ActionLink方法传递的第二个参数只告诉了路由系统要定位到当前View对应的Controller下的Action。Html.ActionLink方法可以使用第三个参数来指定其他的Controller,如下所示: lu V_  
<div> &Ef6'  
    @Html.ActionLink("This targets another controller", "Index", "Admin") 0coRar?+b  
</div> x DX_s:A  
它会自动生成如下链接: o)Nm5g  
<a href="/Admin">This targets another controller</a> B-aJn8>/  
生成带有URL参数的链接 a: [m;  
有时候我们想在连接后面加上参数以传递数据,如 ?id=xxx 。那么我们可以给Html.ActionLink方法指定一个匿名类型的参数,如下所示: Dlo4Wy  
<div>   @Html.ActionLink("This is an outgoing URL", "CustomVariable", new { id = "Hello" }) DP4l %2m0  
</div> !6{b)P  
它生成的Html如下: N;XJMk_ H  
<a href="/Home/CustomVariable/Hello">This is an outgoing URL</a> \DRYqLT`  
指定链接的Html属性 0 +"P 1/  
通过Html.ActionLink方法生成的链接是一个a标签,我们可以在方法的参数中给标签指定Html属性,如下所示: QD*(wj  
<div> vA, t W,  
    @Html.ActionLink("This is an outgoing URL",  "Index", "Home", null, =/F\_/Xw  
        new {id = "myAnchorID", @class = "myCSSClass"}) / S)&dN`  
</div> %dA7`7j  
这里的class加了@符号,是因为class是C#关键字,@符号起到转义的作用。它生成 的Html代码如下: >rzpYc'~w  
<a class="myCSSClass" href="/"id="myAnchorID">This is an outgoing URL</a> ?kM53zbT#  
生成完整的标准链接 SZ"^>}zl=  
前面的都是生成相对路径的URL链接,我们也可以通过Html.ActionLink方法生成完整的标准链接,方法如下: ' C|yUsBC  
<div> ' 8bT9  
    @Html.ActionLink("This is an outgoing URL", "Index", "Home", vt-5 3fa|  
        "https", "myserver.mydomain.com", " myFragmentName", k84JDPu#  
        new { id = "MyId"}, e8<[2J)P&  
        new { id = "myAnchorID", @class = "myCSSClass"}) \m;"KyP+  
</div> } ~#^FFe  
这是Html.ActionLink方法中最多参数的重载方法,它允许我们****请求的协议(https)和目标**器地址(myserver.mydomain.com)等。它生成的链接如下: nH]F$'rtA  
<a class="myCSSClass" id="myAnchorID"    href="https://myserver.mydomain.com/Home/Index/MyId#myFragmentName" >    This is an outgoing URL</a> 3xz{[5<p  
生成URL字符串 b?FTwjV+#  
用Html.ActionLink方法生成一个html链接是非常有用而常见的,如果要生成URL字符串(而不是一个Html链接),我们可以用 Url.Action 方法,使用方法如下: 0q ^dpM  
<div>This is a URL: ;l6tZ]-"  
    @Url.Action("Index", "Home", new { id = "MyId" }) /IN/SZx  
</div> |J#mgA}(  
它显示到页面是这样的: {|J'd+  
4l*4w x""v  
根据指定的路由名称生成URL zbDM+;  
我们可以根据某个特定的路由来生成我们想要的URL,为了更好说明这一点,下面给出两个URL的定义: ]}&f<X  
public static void RegisterRoutes(RouteCollection routes) { s~(iB{-  
    routes.MapRoute("MyRoute", "{controller}/{action}"); 8Dq;QH}  
    routes.MapRoute("MyOtherRoute", "App/{controller}/{action}", new { controller = "Home" }); EO5k?k[*  
} bLysUj5[5  
对于这样的两个路由,对于类似下面这样的写法: p ft6 @ 'q  
@Html.ActionLink("Click me", "Index", "Customer") 3S'j uHT e  
始终会生成这样的链接: }bVWV0Aeim  
<a href="/Customer/Index">Click me</a> oiq7I@Y`x  
也就是说,永远无法使用第二个路由来生成App前缀的链接。这时候我们需要通过另一个方法Html.RouteLink来生成URL了,方法如下: jyt#C7mj-A  
  @Html.RouteLink("Click me", "MyOtherRoute", new {controller="Customer",action="Index" }); e:Zc-  
它会生成如下链接: u,e(5LU  
<a Length="8" href="/App/Customer/Index">Click me</a> # Pulbk8  
这个链接指向的是HomeController下的Index Action。但需要注意,通过这种方式来生成URL是不****的,因为它不能让我们从直观上看到它生成的URL指向的controller和action。所以,非到万不得已的情况才会这样用。 o[bG(qHZ  
在Action方法中生成URL ?5#Ng,8iT  
通常我们一般在View中才会去生成URL,但也有时候我们需要在Action中生成URL,方法如下: uLsGb=m%b  
Public ViewResult MyActionMethod() { *@q+A1P7@  
     Hb3+$vJ^  
    string myActionUrl = Url.Action("Index", new { id = "MyID" }); q9RCXo>Y+1  
    string myRouteUrl = Url.RouteUrl(new { controller = "Home", action = "Index" }); VJX{2$L  
     3:i4DBp,i  
    //... do something with URLs... |#x;}_>7  
    return View(); 66?!"w  
} _=1SR\  
其中 myActionUrl 和 myRouteUrl 将会被分别赋值 /Home/Index/MyID 和 / 。 qwu++9BM  
更多时候我们会在Action方法中将客户端浏览器重定向到别的URL,这时候我们使用RedirectToAction方法,如下: ';x .ry  
public RedirectToRouteResultMyActionMethod() { w0Fi~:b  
    return RedirectToAction("Index"); :".:Wd  
} cWl)ZE<hM  
RedirectToAction的返回结果是一个RedirectToRouteResult类型,它使MVC触发一个重定向行为,并调用指定的Action方法。RedirectToAction也有一些重载方法,可以传入controller等信息。也可以使用RedirectToRoute方法,该方法传入的是object匿名类型,易读性强,如: K,[g<7X5  
publicRedirectToRouteResult MyActionMethod() { h'<}N  
    return RedirectToRoute(new { controller = "Home", action = "Index", id = "MyID"}); Hw5\~!FX  
} GA{>=Q _~  
L`YnrDZK  
13 URL方案最佳实践 y UQ;tTI  
(| Am  
下面是一些使用URL的建议: %5*gsgeI  
最好能直观的看出URL的意义,不要用应用程序的具体信息来定义URL。比如使用 /Articles/Report 比使用 /Website_v2/CachedContentServer/FromCache/Report 好。 |H5){2V>K  
6rL'hB!!]*  
翰用内容标题比使用ID好。比如使用 /Articles/AnnualReport 比使用 /Articles/2392 好。如果一定要使用使用ID(比如有时候可能需要区分相同的标题),那么就两者都用,如 /Articles/2392/AnnualReport ,它看起来很长,但对用户更友好,而且更利于SEO。 !&TbE@Xk  
R! ?8F4G  
对于Web页面不要使用文件扩展名(如 .aspx 或 .mvc)。但对于特殊的文件使用扩展名(如 .jpg、.pdf 和 .zip等)。 \~Zj](#  
bK; -Xcm  
尽可能使用层级关系的URL,如 /Products/Menswear/Shirts/Red,这样用户就能猜到父级URL。 -B$oq8)n*  
~R/7J{Sg  
不区分大小写,这样方便用户输入。 4pkc9\  
F {]:  
揰湫罸用Get和Post。Get一般用来从**器获取只读的信息,当需要操作更改状态时使用Post。 jDcE_55o  
<(?ahO5  
尽可能避免使用标记符号、代码、字符序列等。如果你想要用标记进行分隔,就使用中划线(如 /my-great-article),下划线是不友好的,另外空格和+号都会被URL编码。 7'j?GzaQ+  
:D-d`OyjG>  
不要轻易改变URL,尤其对于互联网网站。如果一定要改,那也要尽可能长的时间保留原来的URL。 .%3bXK+F  
NaG1j+LN  
尽量让URL使用统一的风格或习惯。 TxwZA  
注以上内容来自于:《Pro ASP.NET MVC 4 4th Edition》 I'BoP  
ag14omM-  
附:RouterDebugger安装使用说明 :  }Q`Kg8L  
1.安装 }k duN0  
在程序包控制台中执行命令 ]ss0~2  
PM> Install-Package routedebugger i5|A\Wv"  
安装成功后Web.config文件中会自动加入行 iYHD:cg)~  
<add key="RouteDebugger:Enabled"value="true" /> w@nN3U+  
2.使用 9Q^>.^~^  
MVC3中需要在文件Global.asax中 Application_Start() 中添加代码 PreApplicationStart.Start();  MV**** 以后不需要 3A4?9>g)KU  
如下样式: "X>Z!>  
protected voidApplication_Start() { L/<^uO1  
RegisterRoutes(RouteTable.Routes); KKa"Ba$g  
PreApplicationStart.Start(); //....... } -#Yg B5  
3.禁用 !a ~>;+  
web.config中 @?3u|m |Z  
<add key="RouteDebugger:Enabled" value="true" /> p<`q^D  
改为 * ?rw'  
<add key="RouteDebugger:Enabled" value="false" /> C z4"[C`;  
4.运行效果 b3H~a2"d  
IcoK22/  
RouteDebuger见附件:链接:http://pan.baidu.com/s/1jGqTMXS _w Cp.[3?t  
密码:
本部分内容设定了隐藏,需要回复后才能看到
D+ LeZBJ  
此课程系 我本人的大型商业课程《基于ASP.NET MVC 4 +Knockout.JS+Web API+FluentData+EasyUI 技术实现Web通用商业****发框架》中 项目预备知识部分的 be$wG O=Ts  
课程四《ASP.NET MV**** 核心技术教程》中的《专题三、实例快速上手 - ASP.NET MV**** URL Routing 深入与使用URL最佳实践》 PX(.bP2^Lq  
整套课程预计7月份内上线预****,敬请关注北风官网! uz&CUvos  
此章节视频目录如下: @PZ&/F ^  
+Ys<V  
此处公****前两讲,需要更多资源的同学可以给我留言 jL I(Z  
pTST\0?  
=~?2i)-mC  
.2C}8GGC'  
级别: 北风爱好者

UID: 476548
精华: 0
发帖: 3
威望: 6 点
学点: 1 点
贡献: 0 点
好评: 0 点
学币: 0 个
注册时间: 2014-07-18
最后登录: 2016-11-21
沙发(1楼)  发表于: 2014-07-18 11:43||

别搞得那么经典好不
级别: VIP五级

UID: 270178
精华: 0
发帖: 21
威望: 449 点
学点: 491 点
贡献: 48 点
好评: 0 点
学币: 8 个
注册时间: 2010-11-11
最后登录: 2018-06-07
板凳(2楼)  发表于: 2014-07-22 10:55||

期待MV****+EasyUI+FluentData的新课程!
级别: 北风爱好者

UID: 493021
精华: 0
发帖: 5
威望: 5 点
学点: 0 点
贡献: 0 点
好评: 0 点
学币: 0 个
注册时间: 2014-08-24
最后登录: 2014-08-24
地板(3楼)  发表于: 2014-08-24 23:11||

点赞!!
级别: 北风爱好者

UID: 493021
精华: 0
发帖: 5
威望: 5 点
学点: 0 点
贡献: 0 点
好评: 0 点
学币: 0 个
注册时间: 2014-08-24
最后登录: 2014-08-24
地下室(4楼)  发表于: 2014-08-24 23:16||

感谢分享
级别: VIP三级

UID: 90325
精华: 0
发帖: 3
威望: 2 点
学点: 11 点
贡献: 0 点
好评: 0 点
学币: 0 个
注册时间: 2008-12-07
最后登录: 2016-11-11
下水道(5楼)  发表于: 2014-09-12 15:16||

吐血支持
级别: 北风爱好者

UID: 495644
精华: 0
发帖: 3
威望: 29 点
学点: 36 点
贡献: 3 点
好评: 0 点
学币: 1 个
注册时间: 2014-08-30
最后登录: 2014-09-12
6楼  发表于: 2014-09-12 21:34||

前来认真学习一下了。
级别: 北风工程师

UID: 470673
精华: 0
发帖: 3
威望: 2297 点
学点: 2316 点
贡献: 251 点
好评: 0 点
学币: 19 个
注册时间: 2014-06-29
最后登录: 2015-06-06
7楼  发表于: 2014-09-13 10:42||

好东西值得学习
级别: 北风技术菜鸟

UID: 125044
精华: 0
发帖: 13
威望: 37 点
学点: 36 点
贡献: 2 点
好评: 0 点
学币: 1 个
注册时间: 2009-02-25
最后登录: 2014-09-13
8楼  发表于: 2014-09-13 15:02||

.......
级别: VIP四级

UID: 460455
精华: 0
发帖: 4
威望: 4 点
学点: 13 点
贡献: 0 点
好评: 0 点
学币: 0 个
注册时间: 2014-04-08
最后登录: 2014-10-17
9楼  发表于: 2014-09-29 16:51||

期待课程
上一主题下一主题
«123»Pages: 1/3     Go