ASP.NET MVC5 入门十——添加验证

在本教程:将要在Movie 模型添加验证逻辑确保用户添加或者编辑时数据无误。

DRY(不要重复你自己)

ASP.NET MVC 的核心设计原则之一就是DRY(Don’t Repeat Yourself)。ASP.NET MVC 鼓励你为一个功能或者行为编码一次,在以后需要用到的地方一直复用就可以。这样可以减少编写代码的数量,同时易于维护。

在ASP.NET MVC 和 EF code first 中提供的验证支持就是DRY的好例子。可以只声明一次,然后在任何需要的地方中执行规则。

向Movie模型中添加验证规则

打开 Movies.cs 文件,然后我们来更新Movie类。利用内置的 Required、StringLength、RegularExpression 和 Range Validation 特性添加以下内容:

image

弄完后,我们看看数据库的设计:

image

显然和规格不符,我们来修改它。先生成解决方案,接下来就是利用数据库迁移功能自动生成。命令如下:在软件包管理器控制台输入

  • add-migration DataAnnotations
  • update-database

命令完成时,VS会自动打开一个类文件,我们可以看到其中的Up方法。

public override void Up()
        {
            AlterColumn("dbo.Movies", "Title", c => c.String(maxLength: 60));
            AlterColumn("dbo.Movies", "Genre", c => c.String(nullable: false, maxLength: 30));
            AlterColumn("dbo.Movies", "Rating", c => c.String(maxLength: 5));
        }

Genre 字段不再是可空的,标题和分级都添加了验证条件。我们重新看看数据库的设计:

image

已经改过来了。相关限制条件在源码里面很清楚了,不在细说。

接下来我们看看效果: 进入 localhost:****/Movies/Create

image

 

很明显了,如果没有通过本地客户端的验证,点击Create是不会向服务器提交表单的。

接来下我们看看实现的过程看看在创建视图和创建Action方法时发生了什么

先看看 MoviesController 类中的Create方法:

image

会发现和以前完全一样,没有任何代码上的变化。第一个Create (GET方法)是用来显示初始创建的窗体。第二个(POST版本的)方法先验证是否有错误(ModelStateIsValid)如果存在错误,则重新显示窗体,否则就在数据库中保存新的电影信息。客户端实现校验是通过 javascript 假如我们禁用它看看会发生什么。禁用方法不同浏览器有不同的选项,搜索一下就可以。

禁用 chrome 后,刷新重新输入:

image

没错误提示,我们点击Create

image

很明显,还是有验证,接下来我们输入正确的看看。

image

成功添加进去了,感兴趣的话,在movie.cs 的 POST Create请求添加断点,会发现输入错误的信息,IsValid 是false ,输入正确是True。

如果你添加了断点测试 启用/禁用 JavaScript 2种情况下Create方法的调用会发现,在启用JS的情况下,如果客户端检测到错误是不会调用Action里面POST的Create方法。如果禁用了JS功能,到表单提交到服务器,会通过 ModelState.IsValid 来验证是否数据存在错误。

下一部分:这次改进了 Create 方法的数据验证,下一次就是 Details 和 Delete 方法了。

ASP.NET MVC5 入门六——通过控制器访问数据

首先,我们要添加一个新的控制器,在解决方案资源管理器的Controllers文件夹右键单击,添加,支架。

image

在对话框中,我们选择下图这个: MVC5 控制器带有 读/写 和 视图。

image

名字就叫做:MoviesController 。model class 选择 MVC5DEMO.Models里面的 Movie。

image

image

数据上下文选择 我们的MovieDBContext。然后点击ADD。

image

恩,悲剧了。我们重新修改Models\Movie.cs文件,把Movies 改成Movie。

image

然后ASP.NET MVC5将会自动创建增删查改(CRUD)的方法(action)和视图(views)。是的,一个完整功能的网站就OK了。我们F5运行看看。

image

当然,没有数据,因为我们没有添加任何数据。接下来我们点击 Create New 添加新的电影条目。

添加电影:

image

image

我们来看看代码

继续阅读“ASP.NET MVC5 入门六——通过控制器访问数据”

ASP.NET MVC5 入门七——完善视图

本节中,我们来修改上一篇博文里面系统自动生成的一些代码,下图是运行后的截图:

image

可以看到有很多不足比如:英文太多,日期格式太长等。

我们回到 Models\Movie.cs 添加几句代码:

image

红色框起来的就是增加的部分,using 语句里面的那个 DataAnnotations 我们下载在说。Display 属性指定要显示的字段的名称。原来没有写的话,就是数据库字段的名称,现在我们写上中文了。DataType 指定字段显示的格式,改为日期这样的话时间就不会显示出来,更美观了。

image

在上图左下角,能看到编辑的链接。这个链接是由 index.cshtml 里面的 Html.ActionLink 方法生成的:

image

Html是个Help 类,调用 ActionLink方法来动态生成HTML超链接。ActionLink方法第一个参数指定显示的字符串,第二个参数是 要调用的Action的名称,第三个参数是一个匿名对象。

按照前几篇讲的 /Movies/Edit/3 和 /Movies/Edit?id=3 是等价的。这2个url都是传递给Movies的Edit方法来去执行,同时给出ID。我们去看看这个方法:

image

第一个方法是接受GET请求,第二个是接受HttpPost 。可以看到第二个有个属性【HttpPost】而第一个没有,因为默认就是GET请求。我们注意到第二个Edit方法前面还有一个【ValidateAntiForgeryToken】这个主要是用来防止跨站请求伪造(CSRF)。这个需要和视图里面@Html.AntiForgerToken() 配合使用,会生成一个隐藏的防伪标记。下次写个文章介绍一下。

在上面的GET的Edit 方法,调用实体框架的FIND方法查找时候存在对应id的电影,如果存在就返回给视图,不存在就返回404。下面是视图里面的代码:

image

image

第一行代码,绑定了一个强类型对象。然后接下来利用这个对象,生成 <label> <input>等代码。Html.LabelFor 这个帮助类的方法显示对应的标题,Html.EditorFor 用来生成<input>标签,Html.ValidationMessageFor 就是用来进行验证的。

image

image

生成这么优美的控件,居然一句代码都不用写,真好!

image

这是表单的源文件,可以看到有个隐藏域,存放着上文的那个 RequestVerificationToken 等信息。从这句话可以看出<form action=”/Movies/Edit/2″ method=”post”>当我们点击save后,表单的所有数据都将用post请求发送到 /movies/edit/2 这个url。

回过头来看POST请求的EDIT方法:

image

第一步,判断表单中的模型所有属性时候都可以修改或者更新。如果是有效的,就把实体框架状态改为修改,然后更新到数据库。保存之后,重定向到Index 这个方法,我们就会看到修改后的效果。

在下一个教程:我们来实现搜索功能。

ASP.NET MVC5 入门八——添加搜索

添加一个搜索的方法和搜索的视图

接下来我们添加搜索功能,可以根据电影的标题搜索。

第一步:更新index窗体

在最开始MVC入门说过,MVC一个好处就是关注点分离。当我们需要修改和数据有关的东西,只要去修改Controller 就可以。现在我们去修改MoviesController的Index方法:

image

第一行的LINQ查询用来选择Movies。仅仅是定义了movies代表数据库中的哪些,还没有开始查询。接下来判断查询的字符串是否为空,如果不为空就开始查询标题一样的电影。

s => s.Title 这是一个Lambda 表达式。一般用在基于方法的LINQ查询中,比如上面的Where。

我们现在F5运行一下看看,在地址栏后面加上?searchString=中国

image

image

当然,我们不能让用户一直在地址栏加上那么一串来搜索,普遍做法是,在页面顶部加一个搜索框。看到这个就知道需要改动View部分。我们打开Views\Movies\Index.cshtml 文件,在Create New后面加上如下代码:

image

image

可以发现,我们并没有添加控制器中Index的POST请求,也能正常查询。我们现在添加HttpPost方法

image

image

我们会发现,GET和POST请求的url都是一样的,都是 localhost:xxx/Movies/Index 。假如我们想要把链接发给朋友,让他知道我想看哪些电影,但是这些查询信息却没有在url里面体现,这样不肯能发送给朋友。

解决方法是利用 BeginForm 的重载,有个指定POST请求添加搜索信息到url中。

@using (Html.BeginForm(“Index”,”Movies”,FormMethod.Get))

改成如上的代码,就可以指定用GET方法提交,哪怕你已经有了HttpPost 的方法。

image

接下来:添加更多搜索

继续阅读“ASP.NET MVC5 入门八——添加搜索”

ASP.NET MVC5 入门五——使用LocalDB

在上一篇博文《ASP.NET MVC5 入门四——添加模型》中,我们创建了M模型对应的模型类。最终我们还是要在数据库中创建真实的数据库和表,字段等。对于开发人员,数据库在开发过程中只是需要一些简单的功能,为了开发而安装一个企业版的SQL SERVER太浪费了。还好微软在SQL2012中提供了LocalDB这个新特性,LocalDB也是VS2013默认就安装的。

什么是LocalDB

简单说就是 SQL免费版的SQL Server Express 的轻量级版本。专门为开发人员创立的,易于安装,无需管理,编程语言和T-SQL语言都和正常的数据库一样。同时不耗费系统支援,没有服务,需要时自动启动停止。其他更高级的功能就不说了,够用,免费,好用才是王道。同时LocalDB数据库也能方便的移植到SQL Server。

添加连接字符串

默认情况下,实体框架会查找和对象同名的上下文里面的连接字符串。在项目的根目录下面打开Web.config文件:

image

默认的连接字符串:

image

我们在<connectionStrings>里面添加一个新的连接字符串:

<add name=”MovieDBContext” connectionString=”Data Source=(LocalDB)\v11.0;AttachDbFilename=|DataDirectory|Movies.mdf;Integrated Security=True”
providerName=”System.Data.SqlClient”
/>

可以看出2个连接字符串都很相似。第一个名字是 DefaultConnection 用来控制可以访问该应用程序的成员资格数据库。我们添加自己的Movie.mdf数据库,不使用成员资格数据库。连接字符串的名字必须和DbContext类的名称相匹配。

添加好了之后,就是

下一步:从控制器访问模型中的数据。

ASP.NET MVC5 入门四——添加模型

在本节中,我们将要添加一些类来管理数据库中的信息。MVC入门系类的文章打算建立一个电影网站的示例。在前几篇的博客《Entity Framework学习——了解》,我介绍了新的编程方式 Code First 。所以我们应该添加一些类来作为MVC中的“模型”部分。

添加模型类

在解决方案资源管理器中,右键单击 Models 文件夹,选择添加,然后选择类:

image

输入类名“Movie”。

向类“Movie”添加以下5个属性:

using System;
using System.Collections.Generic;
using System.Linq;
using System.Web;

namespace MVC5DEMO.Models
{
    public class Movie
    {
        public int ID { get; set; }
        public string Title { get; set; }
        public DateTime ReleaseDate { get; set; }
        public string Genre { get; set; }
        public decimal Price { get; set; }
    }
}

从上面的代码可以看出,这个就是用类(class)来表示数据库中的电影(电影数据信息)。类的每个实例对应着数据库中的每行,类的属性对应着数据库中的每列。

在Movie中继续添加代码,我们添加一个 MovieDBContext 类:

image

可以看出,MovieDBContext 继承至 DbContext 。我们需要应用Entity,可以手动加入以下的using System.Data.Entity;语句或者像上图那样,在红色错误处右键选择解析,然后自动添加引用。我们也会发现,有些应用我们不需要,按照下图方式清除未使用的using语句:

image

using System;
using System.Data.Entity;

namespace MVC5DEMO.Models
{
    public class Movie
    {
        public int ID { get; set; }
        public string Title { get; set; }
        public DateTime ReleaseDate { get; set; }
        public string Genre { get; set; }
        public decimal Price { get; set; }
    }

    public class MovieDBContext:DbContext
    {
        public DbSet Movies { get; set; }
    }
}

上面就是最终的代码,清爽许多。这也是我喜欢.NET 胜过JAVA的原因,有个强大的IDE。非常方便,使用起来容易上手。对于中小企业,快速的开发出合格产品才是王道!

下一节:添加数据库连接字符串和LocalDB

ASP.NET MVC5 入门三——添加视图

在上一篇《ASP.NET MVC5 添加控制器》里面我们都是直接返回string 字符串或者html 。这样的话控制器不仅仅做了控制器的工作还做了VIEW(视图)的工作。通常我们不希望控制器直接返回HTML,因为那样子复杂的页面就会非常繁琐,所以我们通常会用一个单独的视图模板文件来辅助生产对应的html响应报文返回客户端。

接下来我们就来看看如何添加视图,先修改上篇博客的那个例子,我们来修改HelloWorldController类,这样才能让view的模板文件来封装要返回的html响应报文。

image

首先,创建一个视图文件夹 HelloWorld 。右键单击(view)——》添加——》新建文件夹。

image

注意名字一定要是 HelloWorld 因为控制器名字是 HelloWorldController 控制器必须是Controller结尾,视图文件夹的名字必须和控制器除了结尾的部分一样。也就是“约定优于配置”的理念。

image

接下来添加一个支架,选择 MVC5 视图—空 不带模型

image

点击添加,弹出的对话框把视图名字(View name)改为 Index 。其他保持默认然后确定。

image

可以看到多了一个 Index.cshtml的文件。在上面的</h2>后面添加如下的html代码:

<p>这是一个View模板,来自于:HelloWorld下面的Index视图!</p>

然后在index.cshtml文件上面右键,选择在页面检查器中查看。当然也可以按下F5进行运行,在浏览器地址栏输入地址进行查看。

image

但浏览器的地址发到服务器的时候,服务器解析给控制器,然后控制器里面只有一句 return View()语句,指定返回一个视图模板文件给浏览器,我们没有写指定什么名字的视图,默认就是Index。

继续阅读“ASP.NET MVC5 入门三——添加视图”

ASP.NET MVC 5 入门二 ——添加控制器

什么是MVC?MVC就是 Models 模型,Views 视图,Controllers 控制器。MVC虽然分为3个部分,但是和三层没有关系,是一种用来开发应用程序的模式具有良好的架构,易于维护和测试。

  • Models:可以理解为是对应用程序的状态、数据以及业务功能的封装。
  • Views:用来显示和用户进行交互的HTML页面。(界面的显示和互动)
  • Controllers:处理传入的请求,完成相应的UI逻辑调用Model里面的相应业务。

下面让我们添加一个控制器类,在资源管理器 项目里面的Controllers 右击。

image

image

接下来去个名字吧,第一个实例一般都喜欢用 helloworld 。

image

我们把代码改成如下,直接返回string字符串在浏览器窗口。

using System;
using System.Collections.Generic;
using System.Linq;
using System.Web;
using System.Web.Mvc;

namespace MVC5DEMO.Controllers
{
    public class HelloWorldController : Controller
    {
        //
        // GET: /HelloWorld/
        public string Index()
        {
            return "这是HelloWorld默认的一个动作(action)";
        }

        //GET:/HelloWorld/Welcome/
        public string Welcome()
        {
            return "这是Welcome的 action 方法";
        }
	}
}

按下F5,在默认的网页地址后面加上Helloworld,如下图:

image

一回车,就会看到我们刚才写的字符串显示在页面上。

ASP.NET MVC 会根据传入的URL调用不同的控制器类(上面例子的HelloWorld)以及里面不同的方法(上面例子的Index、Welcome)。具体依据什么来判断调用的是哪个控制器,哪个方法在App_Start/RouteConfig.cs 文件里面设定。

image

如果直接访问域名,会默认调用 defaults里面的 Home控制器下面的Index方法,后面的ID是传入的参数等数据后面会介绍。

image

所以刚才我们浏览的地址:http://locahost:xxxx/HelloWorld/Welcome MVC默认的映射来解析的话控制器是 HelloWorld ,调用的方法是:Welcome 没有ID这个参数。接下来我们修改为以下代码:

//GET:/HelloWorld/Welcome/
        public string Welcome(string name , int age = 18)
        {
            return HttpUtility.HtmlEncode( "欢迎"+name+",您的年龄是:"+age) ;
        }

给Welcome 方法传入2个参数,年龄默认是18 。接下来我们看看运行结果。

image

以前要获得GET方法传入的参数,需要写一句代码(request),现在MVC自动的帮我们绑定了,多方便!接下来我们把上面的那个代码中的age改成ID,浏览器地址后面添加 /3?name=yan

image

这次的Welcome后面加上了/3 刚好匹配到了ID,剩下的就匹配到了name这个参数。按照这个思路,是不是有办法让地址栏直接匹配到name,这样url里面就没有 ? 这个符号。

我们在App_start\RouteConfig.cs里面添加一个试试:

image

当然,默认的路由规则大多数情况下是很合适的,不需要去修改。接下来会介绍模型绑定,这样就不需要修改路由规则了。

下一步学习:添加视图(View)

ASP.NET MVC 5 入门一——简介、下载安装

本系类教程将要构建ASP.NET MVC 5 Web 应用程序,使用的工具是VS2013 。现在有预览版可以下载了,点击下载VS2013 FOR WEB ISO文件。VS2012 还没用几天呢,2013就要出来,据说数据库有SQL2014的版本了。不带这么玩的!!!

Go

安装完VS2013 for Web 后,打开在起始页(start page)选择新建项目。

image

image

可以发现和VS2012最大的不同,WEB里面只有一个选择,确定后接下来才开始选择其他的:

image

VS2012 还有MVC3 MVC4选择,在VS2013没得选。好了,创建项目后,我们直接按下F5运行。

image

好了,一个网站OK了。拥有 首页、关于、联系方式等页面。更NB的是注册登录也有了。

image

查看了下网页源代码,前端用的是大名鼎鼎的 bootstrap。有时间会专门写这个系类的文章。

下次更新:添加控制器

云中行走——观后感

看完这部电影,还是很有感触的。原因也有和上周刚刚参加完公司的文化培训有光。

文化培训呢,就不具体说了,但是其中关于目标的设定,设定目标的原因看完这部电影更有感触了。主人公小时候看到马戏团表演走钢丝就瞬间喜欢上,但是那时候还没有多远大的目标估计也就马戏团那样的钢丝走一下。一个巧合看到报纸上关于双子楼的描述(世界最高)瞬间就树立了一个目标——我要在双子楼之间走钢丝!而且这个目标是清晰的,远大的(当时来看前无古人),深深印在脑海里的。

企业经营哲学里面关于目标的一句话:乐观构想,悲观计划,乐观执行。在这部电影体验的淋淋尽致!而且主人公一步一步实现这个梦想,先从两棵树的高度开始走钢丝,慢慢变成小塔,在最后的世界高楼。计划的过程也是考虑了很多困难,向有经验的老者请教,最后实施执行的时候遇到各种想不到的问题也能有信心去坚持、相信自己能成功。

另外还有感触的一点是,关于女主人的,和 i do 的广告片段一样,女人放弃了自己的理想成全能男人,可惜男人都不会去好好珍惜。

云中行走真实的事情对比