[ASP.NET MVC 小牛之路]08 - Area 使用


本文摘自PHP中文网,作者黄舟,侵删。

[ASP.NET
MVC 小牛之路]08 - Area 使用

ASP.NET MVC允许使用 Area(区域)来组织Web应用程序,每个Area代表应用程序的不同功能模块。这对于大的工程非常有用,Area 使每个功能模块都有各自的文件夹,文件夹中有自己的Controller、View和Model,但对于管理也增加了一定的难度。

本文目录


创建Area

右键工程选择 添加->区域,弹出如下填写Area的对话框:

785.png

点击添加后,工程目录结构如下:

786.png

和创建一个空MVC工程结构类似,Admin Area 有自己的 Controllers、Models 和 Views 文件夹,不一样的地方就是多了一个 AdminAreaRegistration.cs 文件,这个文件中定义了一个叫 AdminAreaRegistration 的类,它的内容如下:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

namespace MvcApplication1.Areas.Admin {

    public class AdminAreaRegistration : AreaRegistration {

        public override string AreaName {

            get {

                return "Admin";

            }

        }

 

        public override void RegisterArea(AreaRegistrationContext context) {

            context.MapRoute(

                "Admin_default",

                "Admin/{controller}/{action}/{id}",

                new { action = "Index", id = UrlParameter.Optional }

            );

        }

    }

}

系统自动生成的 AdminAreaRegistration 类继承至抽象类 AreaRegistration,并重写

了 AreaName 属性和 RegisterArea 方法。在 RegisterArea 方法中它为我们定义了一个默认路由,我们也可在这个方法中定义专属于Admin Area的的其他路由。但有一点要注意,在这如果要给路由起名字,一定要确保它和整个应用程序不一样。

AreaRegistrationContext 类的 MapRoute 方法和 RouteCollection 类的 MapRoute 方法的使用是一样的,只是 AreaRegistrationContext 类限制了注册的路由只会去匹配当前 Area 的 controller,所以,如果你把在 Area 中添加的 controller 的默认命名空间改了,路由系统将找不到这个controller 。

RegisterArea 方法不需要我们手动去调用,在 Global.asax 中的 Application_Start 方法已经有下面这样一句代码为我们做好了这件事:

1

2

3

4

5

6

7

8

protected void Application_Start() {

    AreaRegistration.RegisterAllAreas();

 

    WebApiConfig.Register(GlobalConfiguration.Configuration);

    FilterConfig.RegisterGlobalFilters(GlobalFilters.Filters);

    RouteConfig.RegisterRoutes(RouteTable.Routes);

    BundleConfig.RegisterBundles(BundleTable.Bundles);

}

调用 AreaRegistration.RegisterAllAreas 方法让MVC应用程序在启动后会寻找所有继承自 AreaRegistration 的类,并为每个这样的类调用它们的 RegisterArea 方法。

注意:不要轻易改变 Application_Start 中注册方法的顺序,如果你把RouteConfig.RegisterRoutes方法放到AreaRegistration.RegisterAllAreas方法之前,Area 路由的注册将会在路由注册之后,路由系统是按顺序来匹配的,所以这样做会让请求 Area 的 Controller 匹配到错误的路由。


Area的运行

在Area中添加controller、view和model和一般的添加是一样的。在这,我们在Admin Area中添加一个名为 Home 的controller,代码如下:

1

2

3

4

5

6

public class HomeController : Controller {

         

    public ActionResult Index() {

        return View();

    }

}

然后我们再为Index Acton添加一个View,代码如下:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

@{

    ViewBag.Title = "Index";

    Layout = null;

}

 

<!DOCTYPE html>

 

<html>

<head>

    <meta name="viewport" content="width=device-width" />

    <title>Index</title>

</head>

<body>

    <p>

        <h2>Admin Area Index</h2>

    </p>

</body>

</html>

运行应用程序,然后将URL定位到/Admin/Home/Index,下面是运行结果:

787.png

到这,我们已经看到,Area中的的工作流程其实就是和根目录下的流程是一样的。但Area并不是一个完全独立的工作空间,我们下面来看看。


Controller的歧义问题

试想一下,如果我们现在在根目录的 Controller 文件夹中也添加一个名为 Home 的 Controller,然后我们通过把URL定位到 /Home/Index,路由系统能匹配到根目录下的 Controller 吗?

在根目录的 Controllers 文件夹中添加好 HomeController 后,为Index添加View,内容随意:

1

2

3

4

5

6

7

...

<body>

    <p>

        <h2>Root Index</h2>

    </p>

</body>

...

路由不改动,我们使用 RouteConfig.cs 文件中系统定义的默认路由:

1

2

3

4

5

6

7

8

9

public static void RegisterRoutes(RouteCollection routes) {

    routes.IgnoreRoute("{resource}.axd/{*pathInfo}");

 

    routes.MapRoute(

        name: "Default",

        url: "{controller}/{action}/{id}",

        defaults: new { controller = "Home", action = "Index", id = UrlParameter.Optional }

    );

}

运行程序,将URL定位到 /Home/Index。结果我们会看到如下错误信息:

788.png

出现这个问题是因为路由系统进行匹配的时候出现了Controller同名的歧义。

当Area被注册的时候,Area中定义的路由被限制了只寻找 Area 中的Controller,所以我们请求 /Admin/Home/Index 时能正常得到 MvcApplication1.Areas.Admin.Controllers 命名空间的 HomeController。然而我们在RouteConfig.cs文件的RegisterRoutes方法中定义的路由并没有类似的限制。

为了解决这个问题,我们需要在RouteConfig.cs文件中定义的路由中加上对应的 namespaces 参数。RouteConfig.cs 中修改后的路由如下:

1

2

3

4

5

6

7

8

9

10

public static void RegisterRoutes(RouteCollection routes) {

    routes.IgnoreRoute("{resource}.axd/{*pathInfo}");

 

    routes.MapRoute(

        name: "Default",

        url: "{controller}/{action}/{id}",

        defaults: new { controller = "Home", action = "Index", id = UrlParameter.Optional },

        namespaces: new[] { "MvcApplication1.Controllers" }

    );

}

运行程序,如下结果说明解决了同名歧义问题:

789.png

添加了 namespaces 参数后,路由系统在对这个路由进行匹配时,优先匹配指定命名空间的controller,如果匹配到则即刻停止查找,如果在指定的命名空间下没有匹配到对应的controller,再按照一般的方式进行匹配。

生成Area URL链接

关于Area的URL链接生成,可以分为这么三种情况:第一种是在当前Area生成指向当前Area的链接;第二种是生成指向其他Area的链接;第三种是在某个Area中生成指向根目录的链接。下面是这三种情况生成链接的方法,使用的路由定义是系统默认的。

如果要在Area中生成当前Area的URL链接,直接用下面的方法就行:

1

@Html.ActionLink("Click me", "About")

它根据当前所在的Area和Controller会生成如下Html代码:

1

<a href="/Admin/Home/About">Click me</a>


如果要生成其他Area的URL链接,则需要在Html.ActionLink方法的匿名参数中使用一个名为area的变量来指定要生成链接的Area名称,如下:

1

@Html.ActionLink("Click me to go to another area", "Index", new { area = "Support" })

它会根据被指定的Area去找路由的定义,假定在Support Area中定义了对应的路由,那么它会生成如下链接:

1

<a href="/Support/Home/Index">Click me to go to another area</a>


如果要在当前Area生成指根目录某个controller的链接,那么只要把area变量置成空字符串就行,如下:

1

@Html.ActionLink("Click me to go to top-level part", "Index", new { area = "" })

相关阅读 >>

asp.net如何利用ashx生成图形验证码的实例

asp.net core实例详解二(环境设置)

解决asp.net网站第一次访问慢的方法

常用的asp.net 技巧总结

关于asp.net如何利用ajaxpro完成前端跟后台交互的实例分析

asp.net如何实现的群发邮件功能详解

详细介绍《asp.net》数据绑定――gridview

详解《asp.net》数据绑定―datalist

使用asp.net mvc使用jsonresult返回json数据实例详解

浅析asp.net使用image/imagebutton空间显示本地图片的方法

更多相关阅读请进入《asp.net》频道 >>




打赏

取消

感谢您的支持,我会继续努力的!

扫码支持
扫码打赏,您说多少就多少

打开支付宝扫一扫,即可进行扫码打赏哦

分享从这里开始,精彩与您同在

评论

管理员已关闭评论功能...