ASP.NET MVC Framework是微软在ASP.NET中所添加的一组类库,这组类库可以使用Model-View-Controller的设计模式来开发ASP.NET的应用程序。它与现有的ASP.NET应用程序并没有冲突,所以两者是可以并发的。ASP.NET MVC Framework被包装在System.Web.Mvc.dll中,并利用ASP.NET Routing来支持动作流以及URL Rewriting的能力,让它可以更贴近Web的发展以及Web 2.0的特性。
ASP.NET MVC Framework的第一个版本于2009年3月17日发布RTM版本,最新的ASP.NET MVC 4.0则是于2012年8月15日正式发布。
ASP.NET MVC Framework对于多数有ASP开发经验的开发人员来说看起来比较不陌生,但对于没有接触过像ASP、PHP、JSP、Perl这些Web开发工具的开发人员来说,相对的不容易入门。
目录
[隐藏]原理[编辑]
ASP.NET MVC是遵循软件模式的Model-View-Controller来发展,其中Model指的是数据或是业务逻辑组件,View是呈现给用户看的信息,而Controller则是接取来自用户的指令与数据,并将Model与View做集成的控制器,当服务器接到对ASP.NET MVC应用程序的要求时,服务器(IIS)会先使用UrlRoutingModule(ASP.NET Routing的HTTP 模块),由它来解析是否有包含ASP.NET MVC应用程序的URL,若有,则会产生一个MvcRouteHandler对象,这个对象会装载运行的必要信息,并且会调用包含在URL中的Controller的Execute方法来运行工作[1]。
Controller对象是基于IController
接口的规则所定义,提供针对HTTP要求做回应的一个运行工具,在ASP.NET MVC中已实现一个默认的类 Controller,提供了必要的基础功能,另外也发展了一个 Controller 工厂,称为 Controller Factory,以IControllerFactory
接口定义,亦提供了DefaultControllerFactory,开发人员可以利用基本的类以及利用它们来派生自己的 Controller 或 Controller Factory 来实现自己的控制器逻辑功能。
Model对象则是为ASP.NET MVC提供数据,不过它没有基类,而是使用.NET Framework一般性的数据结构或是现在的ADO.NET数据对象,像是List、Dictionary、DataTable、DataReader与DataSet等等,当然也可以是自己开发的商业对象,这些数据会通过ASP.NET MVC的ModelBinder工具类来与Controller集成,ModelBinder本身是支持泛型(Generic)的,因为各种类型的数据它都可以使用。在ASP.NET MVC中提供了一个DefaultModelBinder对象,可支持大多数的.NET Framework数据类型,以及数组和已实现像是IList、IDictionary以及ICollection等接口的对象[2]。
Model会在Controller运行动作时,作为一个ActionResult对象方式传回给MvcHandler对象,而这个对象即会指定要显示的View对象,像是下列代码所示:
//
// GET: /Person/
public ActionResult Index()
{
return View(people);
}
//
// GET: /Person/Details/5
public ActionResult Details(Person person)
{
return View(person);
}
//
// GET: /Person/Create
public ActionResult Create()
{
return View();
}
//
// POST: /Person/Create
[AcceptVerbs(HttpVerbs.Post)]
public ActionResult Create(Person person)
{
if (!ModelState.IsValid)
{
return View("Create", person);
}
people.Add(person);
return RedirectToAction("Index");
}
|
View对象以IView
与IViewDataContainer
等接口为主,并且以ASP.NET的各式前端接口为主要输出工具,基于MVC的View弹性化设计考量,以往在ASP.NET Web Form的代码与HTML分离模式将不再存在,而是将代码与HTML混合的方式设计,让开发人员可以更精确的对View进行控制,而目前 ASP.NET MVC 支持的 View 有下列几种[3]:
- .aspx网页,由ViewPage来支持。
- .ascx用户控件,由ViewUserControl来支持。
- .master主版页面,由ViewMasterPage来支持。
每个 View 对象都会内含一个泛型的参数,用来装载要呈现的数据(即Model),然后使用类似下面的方式来呈现数据:
<%@ Master Language="C#" Inherits="System.Web.Mvc.ViewMasterPage" %>
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Strict//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-strict.dtd">
<html >
<head runat="server">
<title><asp:ContentPlaceHolder ID="TitleContent" runat="server" /></title>
<link href="../../Content/Site.css" rel="stylesheet" type="text/css" />
</head>
<body>
<div class="page">
<div id="header">
<div id="title">
<h1>My MVC Application</h1>
</div>
<div id="logindisplay">
<% Html.RenderPartial("LogOnUserControl"); %>
</div>
<div id="menucontainer">
<ul id="menu">
<li><%= Html.ActionLink("Home", "Index", "Home")%></li>
<li><%= Html.ActionLink("About", "About", "Home")%></li>
</ul>
</div>
</div>
<div id="main">
<asp:ContentPlaceHolder ID="MainContent" runat="server" />
<div id="footer">
</div>
</div>
</div>
</body>
</html>
|
技术[编辑]
在ASP.NET MVC架构中,除了Controller、Model与View三个主要部分以外,还包含了许多技术以让这三层得以集成并交互运作。
Controller[编辑]
Controller在ASP.NET MVC应用程序中是负责中控的角色,也是来自客户端HTTP要求的处理核心,因此有许多处理与转向HTTP要求的辅助技术在Controller层次都会使用到。
ASP.NET 路由技术[编辑]
用来过滤客户端要求的URL,并借由定义好的路由表(route table)将要求导向至正确的MVC Controller,并调用Controller中的Execute方法运行,而Execute方法会将HTTP动作以及实际运行的指令交给正确的函数来运行。而通常一个MVC应用程序的URL都会是类似这样的URL格式:
http://127.0.0.1/ControllerName/ActionName/ActionParameters
而MvcRouteHandler会拆解URL,找出目标的Controller,并且将ActionName以及ActionParameters传给Controller中负责的函数(以ActionName来指定)。例如下列的URL会传递给BlogController的GetList方法:
http://127.0.0.1/Blog/GetList
动作与方法直接集成[编辑]
ASP.NET MVC利用了中介数据的技术,直接将方法对应到指定的 HTTP 动词 (GET/POST/PUT/DELETE/HEAD等),MvcHandler会判断要求的类,并将它交给URL中指定的方法来处理。目前MVC Framework可用下列的方式指定(均包含在 HttpVerbs 枚举类型中):
HttpVerbs.Get
HttpVerbs.Post
HttpVerbs.Delete
HttpVerbs.Put
HttpVerbs.Head
将数据模型与展示层直接包装[编辑]
ViewPage
、ViewMasterPage
、ViewUserControl
等展示对象都支持泛型对象,可以直接装载Model数据传递至前端输出,可简化处理Model与View之间集成的动作,只要一个参数就可以将数据传给View:
public ActionResult GetList()
{
return View(BlogDataModel);
}
Model[编辑]
在 ASP.NET MVC 中,Model 相对不设限,可以使用内置的数据结构以及自定义的数据类,也可以是一个商业对象,因此 Model 的弹性相当大,除了前述的数据结构外,微软新发展的一些数据访问方式也可以应用在 Model 中,像是ADO.NET Entity Framework与LINQ to SQL等技术。
另外,MVC在服务端数据验证中,提供了ViewDataDictionary
类,这个类中有一个ModelState属性,内含了ModelStateDictionary
类,开发人员可以利用这个类来控制数据验证的结果,而View中输出验证消息的部分会和此类有关系,例如下列的程序:
[AcceptVerbs(HttpVerbs.Post)]
public ActionResult Create(Person person)
{
if (person.Name.Trim().Length == 0)
{
ModelState.AddModelError("Name", "Name is required.");
}
if (person.Age < 1 || person.Age > 200)
{
ModelState.AddModelError("Age", "Age must be within range 1 to 200.");
}
if ((person.Zipcode.Trim().Length > 0) && (!Regex.IsMatch(person.Zipcode, @"^d{5}$|^d{5}-d{4}$")))
{
ModelState.AddModelError("Zipcode", "Zipcode is invalid.");
}
if (!Regex.IsMatch(person.Phone, @"(((d{3}) ?)|(d{3}-))?d{3}-d{4}"))
{
ModelState.AddModelError("Phone", "Phone number is invalid.");
}
if (!Regex.IsMatch(person.Email, @"^[w-.]+@([w-]+.)+[w-]{2,4}$"))
{
ModelState.AddModelError("Email", "Email format is invalid.");
}
if (!ModelState.IsValid)
{
return View("Create", person);
}
people.Add(person);
return RedirectToAction("Index");
}
在 ASP.NET MVC 2.0 中,新增了一个可以直接让 MVC Framework 针对数据字段进行验证控制的模型,称为 Model Validation,它融合了在 .NET Framework 3.5 SP1 发表的 ASP.NET Dynamic Data Framework 中 Data Annotations (数据记号) 的特性,让开发人员可以只利用标记的方式来运行验证,或是利用自定义的代码来扩充数据记号的验证行为。
using System.ComponentModel.DataAnnotations;
namespace MvcDA {
[MetadataType(typeof(ProductMD))]
public partial class Product {
public class ProductMD {
[StringLength(50),Required]
public object Name { get; set; }
[StringLength(15)]
public object Color { get; set; }
[Range(0, 9999)]
public object Weight { get; set; }
// public object NoSuchProperty { get; set; }
}
}
}
View[编辑]
由于View是直接呈现给用户,因此与用户交互的部分都要由此层处理,包含数据的输出以及以客户端操作为主的回应(例如脚本)等。
HTML 工具类[编辑]
HTML工具类在View中是很重要的输出工具,它内置了辅助产生HTML标签的工具方法,多数的HTML语法都可以利用它来产生,包含像链接(<a>)、窗体(<form>)以及窗体控件等等。HTML工具是以HtmlHelper
类为核心,并配合System.Web.Mvc.Html名字空间的方法,以延伸方法(Extension Method)的方式,让产生HTML的程序就有如调用方法般简单[4]:
<h2>Index</h2>
<table>
<tr>
<th></th>
<th>
Id
</th>
<th>
Name
</th>
</tr>
<% foreach (var person in Model) { %>
<tr>
<td>
<%= Html.ActionLink("Details", "Details", person )%>
</td>
<td>
<%= Html.Encode(person.Id) %>
</td>
<td>
<%= Html.Encode(person.Name) %>
</td>
</tr>
<% } %>
</table>
<p>
<%= Html.ActionLink("Create New", "Create") %>
</p>
数据验证[编辑]
View的HTML工具可以配合Model处理数据验证的结果,在ASP.NET中常用的ValidationSummary在这里也支持,而且MVC的架构让验证信息的输出也更加弹性[5]:
<h2>Create</h2>
<%= Html.ValidationSummary("Create was unsuccessful. Please correct the errors and try again.") %>
<% using (Html.BeginForm()) {%>
<fieldset>
<legend>Fields</legend>
<p>
<label for="Name">Name:</label>
<%= Html.TextBox("Name") %> Required
<%= Html.ValidationMessage("Name", "*") %>
</p>
<p>
<label for="Age">Age:</label>
<%= Html.TextBox("Age") %> Required
<%= Html.ValidationMessage("Age", "*") %>
</p>
<p>
<label for="Street">Street:</label>
<%= Html.TextBox("Street") %>
<%= Html.ValidationMessage("Street", "*") %>
</p>
<p>
<label for="City">City:</label>
<%= Html.TextBox("City") %>
<%= Html.ValidationMessage("City", "*") %>
</p>
<p>
<label for="State">State:</label>
<%= Html.TextBox("State") %>
<%= Html.ValidationMessage("State", "*") %>
</p>
<p>
<label for="Zipcode">Zipcode:</label>
<%= Html.TextBox("Zipcode") %>
<%= Html.ValidationMessage("Zipcode", "*") %>
</p>
<p>
<label for="Phone">Phone:</label>
<%= Html.TextBox("Phone") %> Required
<%= Html.ValidationMessage("Phone", "*") %>
</p>
<p>
<label for="Email">Email:</label>
<%= Html.TextBox("Email") %> Required
<%= Html.ValidationMessage("Email", "*") %>
</p>
<p>
<input type="submit" value="Create" />
</p>
</fieldset>
<% } %>
<div>
<%=Html.ActionLink("Back to List", "Index")