我很早就关注ASP.NET的mvc的,因为最开始是学了Java的MVC,由于工作的原因一直在做.Net开发,最近的几个新项目我采用了MVC做了,我个一直都非常喜欢.Net的MVC。我们为什么使用MVC而不是用WebForm呢?下面就来说说MVC的亮点。由于我最近使用都是MVC5.0和EF6.1,所以下面的所有实例都是基于这两个版本的。
1、创建项目内置了Bootsrap
Bootsrap是一个响应式的UI界面库,能快速的搭建响应式界面,如果没有美工,对界面要求不是很高的话完全可以直接作用,很方便。
Bootsrap的推荐网站
http://getbootstrap.com/
http://www.bootcss.com/
2、url路由控制灵活,对seo友好
- public class RouteConfig
- {
- 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 }
- );
- }
- }
这个是RouteConfig注册默认路由。这个类的静态方法RegisterRoutes是在网站启动的时候调用的。
MVC不像webform那样,一个动态url地址是对应到一个本地的一个aspx文件,而mvc是对应一个Controller(控制器)的里面的一个Action(public方法)。mvc是对应的”/Home/About“对应的就是HomeController的名字为About的Action。这种url地址到Controller的Action的对应关系你完全可以按照你的要求设置,甚至可以配置成.html结尾的伪静态
我之前写的一篇文章可以看一下”自定义route路由机制“
3、视图引擎的灵活
默认情况下创建mvc视图文件.cshtml会自动把这些同一个Controller的Action的视图放在一个文件夹。
对应视图文件结构
mvc视图自带的视图是razor引擎,可以强类型绑定视图,安全,性能方面都有保证。
要指定视图对应的model类型很简单
在Contoller里面用return View(xx)
xx为对应的一个model对象。
视图使用:
@model xxx.Models.xx
这样视图就可以用@Model.字段绑定了。
视图可以像可以定义一个共用的部分Layout(就像Webform的master母版页),头部、页脚、菜单导航等这些共用的html都可以放在Layout里面,对于局部多个地方相同的html,可以用@Html.Action("actionName","controllerName")方法来绑定一个局部视图(就像Webform的用户控件)
- public class NavController : Controller
- {
- [OutputCache(Duration=7200)]
- [ChildActionOnly]
- public PartialViewResult SpecialListMenu()
- {
- return PartialView();
- }
- [OutputCache(Duration = 7200)]
- [ChildActionOnly]
- public PartialViewResult NewsListMenu()
- {
- return PartialView();
- }
- }
Action对应的局部视图也可以定义输出缓存OutputCache单位是分钟,这样下次请求直接在缓存中取出来,提高了程序的效率。 一般对变化不频繁的Action我都这样会加上缓存。加上[ChildActionOnly]表示只能通过视图来引用,不能直接在浏览器访问。
3.1、自定义视图引擎
分享一下AtomicCms里面的自定义视图引擎的代码。
- using System.Collections.Generic;
- using System.Web.Mvc;
- namespace AtomicCms.Web.Core.Mvc
- {
- public class CustomRazorViewEngine : RazorViewEngine
- {
- public CustomRazorViewEngine()
- {
- string[] mastersLocation = new[]
- {
- string.Format("~/skins/{0}/views/{0}.cshtml",
- Utils.SkinName)
- };
- MasterLocationFormats = CustomViewEngineHelper.AddNewLocationFormats(
- new List<string>(MasterLocationFormats),
- mastersLocation);
- string[] viewsLocation = new[]
- {
- string.Format("~/skins/{0}/Views/{{1}}/{{0}}.cshtml",
- Utils.SkinName)
- };
- ViewLocationFormats =
- PartialViewLocationFormats =
- CustomViewEngineHelper.AddNewLocationFormats(new List<string>(ViewLocationFormats),
- viewsLocation);
- }
- public override ViewEngineResult FindView(ControllerContext controllerContext,
- string viewName,
- string masterName,
- bool useCache)
- {
- masterName = CustomViewEngineHelper.OverrideMasterPage(masterName,
- controllerContext);
- return base.FindView(controllerContext,
- viewName,
- masterName,
- useCache);
- }
- }
- }
这样可以实现mvc视图主题,网站可以制作不同风格的主题,每个主题分别绑定视图就可以了。
要使自定义的视图引擎生效还需要在Global.asax加入下面的代码把默认视图引擎禁用。
- ViewEngines.Engines.Clear();
- ViewEngines.Engines.Add(new CustomRazorViewEngine());
4、Model绑定
Action的参数可以是一个个单独的参数,也可以是一个model对象,mvc可以从请求中获取到参数绑定相应的字段到model中去。这样实现了表单提交的时候,自动装配的功能,而不需要类似这样:Request.Form["xx"]获取值赋值给model。
mvc默认的参数绑定是按照以下顺序的。
Request.Form=》RouteData.Values=》Request.QueryString=》Request.Files
假设Action的一个参数id,会从下面按顺序获致id的值,一旦找到就不会再往下面寻找。
1. Request.Form["id"]
2. RouteData.Values["id"]
3. Request.QueryString["id"]
4. Request.Files["id"]
实践开发过程中的一个添加功能,绑定一个model对象,视图代码:
- @model MvcModels.Models.Person
- @{
- ViewBag.Title = "CreatePerson";
- }
- <h2>Create Person</h2>
- @using(Html.BeginForm()) {
- <div>@Html.LabelFor(m => m.PersonId)@Html.EditorFor(m=>m.PersonId)</div>
- <div>@Html.LabelFor(m => m.FirstName)@Html.EditorFor(m=>m.FirstName)</div>
- <div>@Html.LabelFor(m => m.LastName)@Html.EditorFor(m=>m.LastName)</div>
- <div>@Html.LabelFor(m => m.Role)@Html.EditorFor(m=>m.Role)</div>
- <button type="submit">Submit</button>
- }
后台代码
- [HttpPost]
- public ActionResult CreatePerson(Person model) {
- return View("Index", model);
- }
这样前台的表单的值会对应到model的相应的字段里面。当然也可以指定哪些字段要绑定
- public ActionResult AddSummary(
- [Bind(Include = "HomeAddress", Exclude = "Country")]AddressSummary summary)
- {
- //do something
- return View(summary);
- }
MVC的校验非常的好,自带非空检验,类型检验,也可以写复杂的正则表达式。
[Requred]
Public string Name{get;set;}
以上表示Name字段必填。
4.1自定义Binder绑定
使用IModelBinder接口自定义一个购物车的Binder类
- public class CartModelBinder : IModelBinder
- {
- private const string sessionKey = "Cart";
- public object BindModel(ControllerContext controllerContext,
- ModelBindingContext bindingContext)
- {
- //从Session读取购物车对象
- Cart cart = (Cart)controllerContext.HttpContext.Session[sessionKey];
- if (cart == null)
- {
- cart = new Cart();
- controllerContext.HttpContext.Session[sessionKey] = cart;
- }
- return cart;
- }
- }
在Global.asax里面的Application_Start方法加入代码为Model绑定集合加入上面自定义的CartModelBinder类。
ModelBinders.Binders.Add(typeof(Cart), new CartModelBinder());
这样以后的Action的Cart对象如:
- public ViewResult Summary(Cart cart)
- {
- return View(cart);
- }
就会自动绑定,也就是从Session里面取key为Cart的对象。
5、控制器灵活
mvc的Controller拓展也很灵活
上面就是MVC框架程序的执行流程,上面图中的ControllerFactory,Controller,Action Invoker都可以完全自定义拓展。
我们创建的Controller是其实默认继承System.Web.Mvc.Controller的,这是一个抽象类,其实它为我们提供了很多基础实现
它的很多方法都定义为虚方法,所以如果我们要实现自己个性化的东西也可以重写里面的方法。
5.1、自定义一个ControllerFactory
- using System;
- using System.Web.Mvc;
- using System.Web.Routing;
- using System.Web.SessionState;
- using ControllerExtensibility.Controllers;
- namespace ControllerExtensibility.Infrastructure {
- public class CustomControllerFactory: IControllerFactory {
- public IController CreateController(RequestContext requestContext,
- string controllerName) {
- Type targetType = null;
- switch (controllerName) {
- case "Product":
- targetType = typeof(ProductController);
- break;
- case "Customer":
- targetType = typeof(CustomerController);
- break;
- default:
- requestContext.RouteData.Values["controller"] = "Product";
- targetType = typeof(ProductController);
- break;
- }
- return targetType == null ? null :
- (IController)DependencyResolver.Current.GetService(targetType);
- }
- public SessionStateBehavior GetControllerSessionBehavior(RequestContext
- requestContext, string controllerName) {
- return SessionStateBehavior.Default;
- }
- public void ReleaseController(IController controller) {
- IDisposable disposable = controller as IDisposable;
- if (disposable != null) {
- disposable.Dispose();
- }
- }
- }
- }
CustomControllerFactory职责是根据请于的获取到路由信息来创建相应的Controller对象
要使自定义的CustomControllerFactory生效还需要在在Application_Start()方法中加入注册代码
- ControllerBuilder.Current.SetControllerFactory(new CustomControllerFactory());
6、AOP面向方面编程
Java的Spring框架的AOP(面向方面编程)很强大,AOP的优点是大大的降低了软件模块的耦合性,提高了代码的复用和维护性。
ASP.NET MVC有各种Filter过滤器,就相当于AOP的技术。可以把应用于身份验证,日志记录,异常处理,这样核心业务只关心自己的逻辑代码就是了,最后的代码不会参杂有业务代码身份验证、日志相关的代码。
我这之前写了一篇介绍mvc的aop文章,AOP实践--利用MVC5 Filter实现登录状态判断
7、IOC控制反转
.Net方面的IOC框架是也是不少的主流的有Autofac、Castle Windsor、Unity、Spring.NET、StructureMap、Ninject等。MVC使用这些框架也很好集成。有的都不用自己写IOC框架与MVC集成代码了。像Autofac有MVC5和MVC2、3、4都有现有集成代码,如下图。
直接安装就可以在自己的MVC项目中用了,如果你用的IOC框架没有在nuget中找到MVC集成包,自己写有也很容易。
具体请看我这前的文章IOC实践--用Autofac实现MVC5.0的IOC控制反转方法
8、单元测试容易
你可以不通过Web服务器IIS之类的来测试Action和Controller,利用moq框架很容易的就mock模拟出真实的Web请求。
mvc的Controller和Action方法都可以很方便的进行单元测试。
你如果要测试一个Action方法的返回值,你都可以不用解析任何的HTML的。 只需要监视Action的返回值ActionResult类型的对象。
不用模拟用户请求,MVC框架的model绑定对象,做为Action方法的参数,然后返回相应的结果 。最简单的测试就是你传入具体的参数直接调
Action方法,就可以了。比如要测试一个Action是真能返回一个指定的View
- public ViewResult Index() {
- return View("Homepage");
- }
测试代码:
- using System;
- using Microsoft.VisualStudio.TestTools.UnitTesting;
- using ControllersAndActions.Controllers;
- using System.Web.Mvc;
- namespace ControllersAndActions.Tests {
- [TestClass]
- public class ActionTests {
- [TestMethod]
- public void ViewSelectionTest() {
- // 创建要测试的 Controller
- ExampleController target = new ExampleController();
- // 调用Action
- ViewResult result = target.Index();
- // 判断测试结果
- Assert.AreEqual("Homepage", result.ViewName);
- }
- }
- }
9、开源
MVC是开源的框架,而NuGet是一个VS很有用的一个包管理工具,上面有很多有用的类库,搜mvc出现下面的结果。
可以看到很多有用的类库,分页(PagedList.Mvc),Grid.Mvc。
9、entityframework完美配合
MVC5项目添加Controller可以选择”包含视图的MVC5控制器(使用Entity Framework)“,这样一个Model的增、删、改、查这些相关的View和Action自动给你生成好了,只需要根据自己需要改一下基本上就能用了。这样大大的提高的开发速度,对于开发管理后台这个太好用了。
10、总结
总结:我之前一直都觉得.net是入门比较容易,因为很多东西微软都给你封装好了,要成为高手比较难,不像java的好多开源的框架,从中可以学到好多的软件架构和设计模式这类的东西,例如,面向接口编程,AOP,IOC这类的。但是伴随着.Net的相关东西开源,比如MVC、EF、甚至现在.NET Framework也开源了,研究的越来越多,这些.net现在也可以,我上面写就可以看到MVC的每个部分都很灵活,完全可以根据自己的需要重写,订制。建议想成为高手或架构师,完全有必要仔细研究MVC的源码和原来,因为从里面可以学到很多设计模式、软件架构、软件设计相关的技巧,对于提升我们的技术能力很有帮助。