ASP.NET MVC中对数据进行排序的方“.NET研究”法

  本系列是讲解如何在asp.net mvc中对数据进行展示、排序、分页等的系列文章。在上周的文章中,一步一步教会了大家如何使用ASP.NET MVC框架去的展示数据。在上周的文章中,我们先用Visual Studio创建了一个新的ASP.NET MVC应用程序,接着连接到了Northwind数据库,并展示了如何使用微软的LINQ-SQL的工具去访问数据库中的数据,接着指导如何去实现视图层去展示产品信息及如何设计控制器。

  本文是在上一篇文章的例子基础上,展示了如何去实现数据的双向排序。如果你是已经熟悉ASP.NET WebForm开发的开发者,你应该知道,在GridView控件中可以很简单的通过点击就能实现排序。但可惜的是,在ASP.NET MVC中实现排序的功能并不是那么简单,但工作量也没有显着增多。在ASP.NET MVC中,我们能更多地控制网格和排序的界面布局和标记,以及通过何种机制来实现排序。以往使用GridView控件时,排序是通过将参数以POSTBACK的形式回传到后台,以决定用什么样的列进行排序以及是以升序或降序-排列,回传的参数作为隐藏表单域提交。在本文中,我们将使用查询字符串参数来指定排序参数,这意味着排序的顺序可以被搜索引擎搜索到,能通过电子邮件发送给同事,还能做很多GridView内置排序功能不能实现的事情。

  与上一篇文章一样,本文提供了分步的指导说明,包括一个完整的可以工作的代码例子,在文章末尾可以下载。

  步骤0:一个简要指南

  本文将介绍如何实现双向的排序,并假定读者已经阅读掌握了上一篇文章介绍的内容。

  在上一篇文章中,我们可以通过ASP.NET MVC中的URL转发功能,以下面的地址形式访问产品的首页:

  www.yoursite.com/Products/Index(可以简写成www.yoursite.com/Products)。本文中,我们将用下面的URL去访问要排序的内容:

  www.yoursite.com/Products/Sortable?sortBy=ColumnNameascending=true|false

  具体的一些例子如下:

  /Products/Sortable- 这表示按默认顺序排列产品。当没指定排序的列时,默认按产品名称的字母顺序(如升序)排列。换句话说,如果SortBy参数没有提供,按产品名称排序,如果不提供ascending参数,按递增顺序排列。

  /Products/Sortable?sortBy=UnitPrice 按单价的升序排序产品。

  /Products/Sortable?sortBy=UnitPriceascending=false – 按UnitPrice列降序排序(即从最昂贵的到价格最便宜的)。

  跟使用GridView控件一样,在点击网格中标题行中的列名时可以进行。但不同于GridView的是,我们每次点列的这些标题,是以超链接的形式实现的,并且带有参数,比

  如表格中有价格这个列,当第一次点击列名时,将以

  www.yoursite.com/Products/Sortable?sortBy=UnitPriceascending=true的形式发送链接到后端,请注意的是,在网格中显示的列名,不一定跟在URL中sortBy查询字符串参数中传递的名称是一样的。sortBy 参数提供的是在数据库中的列名,两者并不要求一定相同。

  步骤1:创建指定的视图模型

  在上一篇文章的演示中,我们使用了产品的集合作为其实体模型(以NorthwindDataContext去命名)。这对简单的网格来说是可以的,但对于要排序的数据表格,需要知道一点的不仅仅是产品的集合,还要视图层方面知道哪些列的数据需要进行排序,是按升序或降序排序,如果用户要点的列已经按某一个顺序已排序的话,则此时会按原来的顺序排序(假设某列已经是按升序排列,用户点标题一次,则倒过来按降序排列,再点一次,又按升序,如此类推)。

  为此,我们添加一个新类,这些类被称为特定视图服务的模型,打开上一篇文章中已经实现的应用程序,在Models文件夹下添加一个名为ProductGridModel.cs 的文件,代码如下:


namespace Web.Models
{
public class ProductGridModel
{
// Data properties
public IEnumerable Product Products { get; set; }

// Sorting-related properties
public string SortBy { get; set上海徐汇企业网站制作n style="color: #000000;">; }
public bool SortAscending { get; set; }
public string SortExpression
{
get
{
return this.SortAscending ?上海企业网站制作pan> this.SortBy + " asc" : this.SortBy + " desc";
}
}
}
}

  该ProductGridModel类定义了一个产品属性,它是一个集合类的属性,用来显示产品集合,同时也有三种排序相关的属性:

  SortBy

  在数据库中用来排序的数据列名称

  SortAscending

  一个布尔值,指示是否用升序排序数据

  SortExpression

  只读属性,返回一个排序字符串,其构造为SortBy和SortAscending值的组合。

  例如,如果SortBy分为UnitePrice和SortAscending是true,SortExpression的值为

   UnitedPrice asc。

  如果SortBy设置为Discontinued 和SortAscending是false,则SortExpression返回Discontinued desc 。

  步骤2:创建Sortable 的Action方法

  在上一篇教程中,我们创建了一个名为ProductsController的控制器,其中有一个叫index的action和一个辅助属性DataContext,本文中,我们将添加一个新的action方法到控制器中,并命名为Sortable,当有来自如www.yoursite.com/Products/Sortable的请求时,则执行该排序方法。

  ASP.NET MVC中实现了自动参数绑定,来自URL或其他的参数请求,将被映射到执行的实际的action中去。例如,如果你在控制器action中定义了一个输入参数,名为sortBy,则MVC框架将搜索传入的请求的参数,看是否有任何具有相同名称的参数(在这里,一个参数可能是一个提交表单域,一个查询字符串参数或路由参数。)如果找到一个匹配,则自动把参数的值得赋给action中定义的参数。

  下面是其实际代码:


public class ProductsController : Controller

  {

  ...

// GET: /Products/Sortable?SortColumn=columnNameAscending=true|false

  public ActionResult Sortable(string sortBy = "ProductName", bool ascending = true)

  {

  var model
= new ProductGridModel()

  {

  SortBy
= sortBy,

  SortAscending
= ascending

  };

  model.Products
= this.DataContext.Products.OrderBy(model.SortExpression);

return View(model);

  }

  }

  请注意,action中接受两个输入参数:sortBy和ascending。任何来自URL的请求,只要符合这两个参数的名称的,其值得都会被自动匹配传入到该action中去, 也就是说,如果有人访问www.yoursite.com不指定查询字符串参数,则默认按照产品的名称进行升序排序。

  Sortable的action首先创建一个新的ProductGridModel实例,命名为model,并且对model的SortBy和SortAscending属性进行赋值,接着,Sortable action获得了产品的集合(this.DataContext.Products),并调用其中的OrderBy方法进行排序,而排序的参数表达式正好是SortExpression。最后,将model模型返回给一个强类型的视图,。

  如果您熟悉使用LINQ,你会觉得我在这里使用的OrderBy方法有点奇怪,你可能会用LINQ中的lambda表达式如下这样写,如:this.DataContext.Products.OrderBy(p =p.ProductName)。

  OrderBy方法,我使用的是不标准的LINQ,相反是微软的动态LINQ库中的方法

  (http://weblogs.asp.net/scottgu/archive/2008/01/07/dynamic-linq-part-1-using-the-linq-dynamic-query-library.aspx),它是一个库允许使用字符串做为查询参数。本文附件中提供了一个名为Dynamic.cs的文件(它在HelperClasses文件夹中),其中包含了OrderBy扩展方法签名,大家可以去学习一下。 

  步骤3:创建视图

  鼠标右键点控制器中的Sortable action,在弹出的菜单中选择添加视图选项。从添加视图对话框中,选中Create a strongly-typed view 复选框,然后从View data class 下拉选择Web.Models.ProductGridModel,点击确定。这时会创建视图Sortable.aspx,如下图:

  接下来,修改代码如下:


table class="grid"
tr
thProduct/th
thCategory/th
thQty/Unit/th
上海闵行企业网站制作an style="color: #800000;">thPrice/th
thDiscontinued/th
/tr% foreach (var item in Model.Products) { %

tr
td class="left"%: item.ProductName %/td
td class="left"%: item.Category.CategoryName %/td
td class="left"%: item.QuantityPerUnit %/td
td class="right"%: String.Format("{0:C}", item.UnitPrice) %/td
td
% if (item.Discontinued) { %
img src="%=Url.Content("~/Content/cancel.png") %" alt="Discontinued" title="Discontinued" /
% } %
/td
/tr

% } %/table

  上面的代码跟上一篇文章中你看到的基本没什么区别,唯一的区别,其实是在foreach语句中,在上一篇文章中,是:


  foreach(var item in Model) { ... },而本文中,其模型变成了ProductGridModel实例,其产品属性包含产品的集合。因此,在foreach循环遍历Model.Products。运行后,你会发现访问页面时默认的是按产品的升序排列的:

  现在网格的列标题是文本,等下我们会将其修改为超级链接,但现在我们也可以马上在浏览器中,通过输入的方法体验下了,举例来说,如果你输入:

  www.yoursite.com/Products/Sortable?sortBy=UnitPriceascending=false- 你应该看到按 照价格从高到低排列的产品。

  在网格中添加排序链接

  此时,用户无法直观地去排序,除非他在浏览器中象上面的方法输入查询字符串,因此我们在每个网格的标题行中提供超链接指向适当的网址。乍一看,这可能会像一个简单的任务,比如产品名称列,其url应该是

  Products/Sortable?sortBy=ProductNameascending=true,而价格列,则应该是

  Products/Sortable?sortBy=UnitPriceascending=true,如此类推,归纳一下,其URL的连接方式应该象:

  Products/Sortable?sortBy=ColumnNameascending=false,同时,还应该增加一个象图标这样的指示,让用户可以清楚地看到当前排序的情况。

  要做到这一点,需要创建一个局部视图。一个局部视图看上去跟WebForms模型中的用户控件的概念差不多。总之,这是一可以重用的视图。

  在解决方案资源管理器中的Views/Shared目录下,鼠标右键点击,在弹出的菜单中选择添加视图,将其命名为SmartLink,并选中下面的两个复选框,然后选择Web.Models.ProductGridModel作为其view data class,如下图。

  单击确定后,Visual Studio将创建一个新的局部视图(SmartLink.ascx)。其中局部视图只包含下列标记:


%@Control
Language="C#" Inherits="System.Web.Mvc.ViewUserControlWeb.Models.ProductGridModel" %

  我们现在来学习如何将局部视图添加到正常的视图中去,这可以使用下列Html.RenderParial方法之一:


% Html.RenderPartial("partialViewName"); %
% Html.RenderPartial("partialViewName", viewData); %
% Html.RenderPartial("partialViewName", model); %
% Html.RenderPartial("partialViewName", model, viewData); %

  可选的参数是一个ViewDataDictionary的ViewData对象。如果提供,可以通过

  ViewData["name"].去访问这些值。

  当要展示局部视图的时候,我们还需要将ProductGridModel和一些额外的信息传进来,下面的代码展示了对于产品名称一列这个表头,我们是如何通过局部视图去产生的,注意的是,我们通过ViewDataDictionary可以设置表头中显示的列的名称以及该列实际对应的是数据库中的列名:


% Html.RenderPartial("SmartLink", Model, new ViewDataDictionary {
{ "ColumnName", "ProductName" }, { "DisplayName", "Product" } }); %

  这里设置了列名是ProductName,而显示在表头中的列名是Product。同理,其他表头列也是这样的处理,就可以把Model传递到部分视图中来了。


table class="grid"
tr
th% Html.RenderPartial("SmartLink", Model, new ViewDataDictionary { { "ColumnName", "ProductName" }, { "DisplayName", "Product" } }); %/th
th% Html.RenderPartial("SmartLink", Model, new ViewDataDictionary { { "ColumnName", "Category.CategoryName" }, { "DisplayName", "Category" } }); %/th
th% Html.RenderPartial("SmartLink", Model, new ViewDataDictionary { { "ColumnName", "QuantityPerUnit" }, { "DisplayName", "Qty/Unit" } }); %/th
th% Html.RenderPartial("SmartLink", Model, new ViewDataDictionary { { "ColumnName", "UnitPrice" }, { "DisplayName", "Price" } }); %/th
th% Html.RenderPartial("SmartLink", Model, new ViewDataDictionary { { "ColumnName", "Discontinued" }, { "DisplayName" , "Discontinued" } }); %/th
/tr

...
/table

  现在我们可以看到,从局部视图中可以通过Model属性访问在Sortable视图中的任何值了,访问的方法是通过ViewData[name]即可。还要记得,我们之所以要在这里使用局部视图,其目的为某个特定的列生成超级链接。

  我们接下来要判断某个列是否要升序还是降序排列,下面的代码创建了isDescending变量,这个值只有当要排序的列(由Model.SortBy产生)和数据库中的列名相同(由ViewDate[ColumnName]产生)时,并且当前的排序为升序(Model.SortAscending)时,其值才为True。


var isDescending = string.CompareOrdinal(Model.SortBy,
ViewData["ColumnName"].ToString()) == 0 Model.SortAscending;

  同时,我们最好通过设置CSS样式,告诉用户当前的排序方向,因此我们在 CustomStyles.css的CSS类定义了两个CSS类-sortAsc和sortDesc,增加了一个向上或向下箭头,可以用如下代码去判断:


f (string.CompareOrdinal(Model.SortBy, ViewData["ColumnName"].ToString()) == 0)
{
if (Model.SortAscending)
htmlAttributes.Add("class", "sortAsc");
else
htmlAttributes.Add("class", "sortDesc");

  最后,完成的局部视图代码如下:


%
var isDescending = string.CompareOrdinal(Model.SortBy, ViewData["ColumnName"].ToString()) == 0 Model.SortAscending;

var routeData
= new RouteValueDictionary { { "sortBy", ViewData["ColumnName"].ToString() }, { "ascending", !isDescending } };

var htmlAttributes
= new Dictionarystring, object();
if (string.CompareOrdinal(Model.SortBy, ViewData["ColumnName"].ToString()) == 0)
{
if (Model.SortAscending)
htmlAttributes.Add("class", "sortAsc");
else
htmlAttributes.Add("class", "sortDesc");
}
%%: Html.ActionLink(
ViewData["DisplayName"].ToString(), // Link Text
Html.ViewContext.RouteData.Values["action"].ToString(), // Action
Html.ViewContext.RouteData.Values["controller"].ToString(), // Controller
routeData, // Route data上海闵行企业网站设计与制作>
htmlAttributes // HTML attributes to apply to hyperlink
)
%

  Html.ActionLink方法会向浏览器端产生超链接的HTML。它的第一个参数为要显示的文字超链接的标题,这里通过ViewData[displayname]读取。

  第二个和第三个参数指定了action和控制器去生成链接,我们使用Html.ViewContext.RouteData.Values[action].ToString

  和Html.ViewContext.RouteData.Values[controller].ToString()来获得当前请求的action和控制器,而避免了硬编码。

  第四个输入参数指定了路由的数据,这是我们用RouteValueDictionary字典的形式,分别向sortBy和ascending参数传入值。

  最后的参数设定了生成链接的HTML的CSS样式。

  当运行程序后,默认是按产品名称升序排序的,如下图。

  当点产品标题列旁边的小箭头时,会向控制器发出如

  Products/Sortable?sortBy=ProductNameascending=False的请求,于是产品名称按降序排序了,如下图:

  本文的代码可以在:http://aspnet.4guysfromrolla.com/code/GridDemosMVC.zip下载得到。

时间: 2024-11-02 02:59:07

ASP.NET MVC中对数据进行排序的方“.NET研究”法的相关文章

一起谈.NET技术,ASP.NET MVC中对数据进行排序的方法

本系列是讲解如何在asp.net mvc中对数据进行展示.排序.分页等的系列文章.在上周的文章中,一步一步教会了大家如何使用ASP.NET MVC框架去的展示数据.在上周的文章中,我们先用Visual Studio创建了一个新的ASP.NET MVC应用程序,接着连接到了Northwind数据库,并展示了如何使用微软的LINQ-SQL的工具去访问数据库中的数据,接着指导如何去实现视图层去展示产品信息及如何设计控制器. 本文是在上一篇文章的例子基础上,展示了如何去实现数据的双向排序.如果你是已经熟

asp.net mvc中,ip怎么传入sql数据库库

问题描述 asp.net mvc中,ip怎么传入sql数据库库 asp.net mvc中,我在control中获取了ip地址,但是怎么传入sql数据库呢?新手,只知道怎么提交用户填写的表单里的数据... 解决方案 假设你用的是ef一类的框架,那么xxxDataEntites db = new xxxDataEntites();db.表.Add(new 实体 { ip = 你获得的ip });db.SaveChanges(); 解决方案二: ajax啊,你给他绑到用户点击的地方 解决方案三: 在数

包含在ASP.NET MVC中的过滤器

在深入研究如何编写过滤器之前,首先看看包含在ASP.NET MVC中的过滤器. ASP.NET MVC包括了如下3种即开即用的动作过滤器: Authorize:该过滤器用于限制对控制器或控制器动作的访问. HandleError:该过滤器用来指定一个处理异常的动作,这个异常是从动作方法的内部抛出的. OutputCache:该过滤器用来为动作方法提供输出的缓存. 接下来将依次深入讨论这3个过滤器. 1  Authorize AuthorizeAttribute是包含在ASP.NET MVC中默认

ASP.NET MVC中的视图生成简介

在 ASP.NET MVC 中,我们将前端的呈现划分为三个独立的部分来实现,Controller 用来控制用户的操作,View 用来控制呈现的内容,Model 用来表示处理的数据. 从控制器到视图 通常,在 Controller 中,我们定义多个 Action ,每个 Action 的返回类型一般是 ActionResult,在 Action 处理的最后,我们返回对于视图的调用. public ActionResult Index() {    return this.View(); } 默认情

ASP.NET MVC中的ActionFilter是如何执行的?

在ASP.NET MVC中的四大筛选器(Filter),ActionFilter直接应用在某个Action方法上,它在目标Action方法执行前后对调用进行拦截以执行一些额外的操作.这是一种典型的AOP式的设计,如果我们需要在执行某个Action方法的前后执行一些操作,可以通过定义ActionFilter来实现.本篇文章主要讲述多一个应用到相同Action方法上的ActionFilter的执行机制. 一.ActionFilter ActionFilter允许我们在目标Action方法执行前后对调

ASP.NET MVC中你必须知道的13个扩展点

ScottGu在其最新的博文中推荐了Simone Chiaretta的文章13 ASP.NET MVC extensibility points you have to know,该文章为我们简单介绍了ASP.NET MVC中的13个扩展点.Keyvan Nayyeri(与Simone合著了Beginning ASP.NET MVC 1.0一书)又陆续发表了一些文章,对这13个扩展点分别进行深入的讨论.我将在 以后的随笔中对这些文章逐一进行翻译,希望能对大家有所帮助. ASP.NET MVC设计

在 Asp.NET MVC 中使用 SignalR 实现推送功能

原文http://www.cnblogs.com/kesalin/archive/2012/11/09/signalr_push.html 在 Asp.NET MVC 中使用 SignalR 实现推送功能 罗朝辉 ( http://www.cnblogs.com/kesalin/ ) CC许可,转载请注明出处   一,简介 Signal 是微软支持的一个运行在 Dot NET 平台上的 html websocket 框架.它出现的主要目的是实现服务器主动推送(Push)消息到客户端页面,这样客户

在ASP.NET MVC中使用“RadioButtonList”和“CheckBoxList”

在<为HtmlHelper添加一个RadioButtonList扩展方法>中我通过对HtmlHelper和HtmlHelper<Model>的扩展使我们可以采用"RadioButtonList"的方式对一组类型为"radio"的<input>元素进行操作.昨天对对此进行了一些改进,并将"CheckBoxList"的功能添加进来.[源代码从这里下载] 一.有何特别之处? 和我的很多文章一样,旨在提供一种大体的解决

Asp.Net MVC 分页、检索、排序整体实现

原文:Asp.Net MVC 分页.检索.排序整体实现     很多时候需要这样的功能,对表格进行分页.排序和检索.这个有很多实现的方式,有现成的表格控件.用前端的mvvm,用户控件.但很多时候看着很漂亮的东西你想进一步控制的时候却不那么如意.这里自己实现一次,功能不是高大全,但求一个清楚明白,也欢迎园友拍砖.前端是bootstrap3+jPaginate,后台基于membership.没什么难点.     先上效果图.            分页其实就是处理好 每页项目数.总项目数.总页数.当