作者:Tom Dykstra和Rick Anderson
Contoso 大学示例 web 应用程序演示如何使用 Entity Framework Core 和 Visual Studio 创建 ASP.NET Core MVC web 应用程序。 想要获取有关系列教程的信息,请参阅第一个教程。
在前面的教程,你可以实现一组的用于学生实体的基本 CRUD 操作网页。 在本教程将向学生索引页添加排序、 筛选和分页功能。 你还将创建具有简单分组功能的页面。
下图显示你完成本教程后相关页面的样子。 列标题时一个链接,用户可以单击它使数据按该列排序。 反复单击列标题在升序排列和降序排列之间切换。
为了添加排序学生索引页,你将更改学生控制器中的Index
方法并向学生索引视图添加相关的代码。
在StudentsController.cs,用以下代码替换Index
方法:
public async Task Index(string sortOrder)
{
ViewData["NameSortParm"] = String.IsNullOrEmpty(sortOrder) ? "name_desc" : "";
ViewData["DateSortParm"] = sortOrder == "Date" ? "date_desc" : "Date";
var students = from s in _context.Students
select s;
switch (sortOrder)
{
case "name_desc":
students = students.OrderByDescending(s => s.LastName);
break;
case "Date":
students = students.OrderBy(s => s.EnrollmentDate);
break;
case "date_desc":
students = students.OrderByDescending(s => s.EnrollmentDate);
break;
default:
students = students.OrderBy(s => s.LastName);
break;
}
return View(await students.AsNoTracking().ToListAsync());
}
此代码从 URL 中的查询字符串中接收sortOrder
参数。 ASP.NET Core MVC 提供的查询字符串作为参数传递给的操作方法。 “Name”或”Date”,后面可以选择性跟用于指定降序顺序的下划线和”desc”构成参数字符串。 默认排序顺序为升序。
第一次请求索引页时,没有任何查询字符串。 学生按姓氏升序显示也就是switch
语句中的缺省值中的排序方式。 当用户单击列标题的超链接,将向Index
方法提供相应的sortOrder
查询字符串。
视图使用ViewData
元素中两个元素 (NameSortParm 和 DateSortParm) 对应的查询字符串值配置列标题超链接。
public async Task Index(string sortOrder)
{
ViewData["NameSortParm"] = String.IsNullOrEmpty(sortOrder) ? "name_desc" : "";
ViewData["DateSortParm"] = sortOrder == "Date" ? "date_desc" : "Date";
var students = from s in _context.Students
select s;
switch (sortOrder)
{
case "name_desc":
students = students.OrderByDescending(s => s.LastName);
break;
case "Date":
students = students.OrderBy(s => s.EnrollmentDate);
break;
case "date_desc":
students = students.OrderByDescending(s => s.EnrollmentDate);
break;
default:
students = students.OrderBy(s => s.LastName);
break;
}
return View(await students.AsNoTracking().ToListAsync());
}
这两个语句都使用了三目运算符。 第一个语句指如果sortOrder
参数为 null 或为空则 NameSortParm 应设置为”name_desc”; 否则,它应设置为一个空字符串。 这两个语句使试图能够如下所示设置列标题的超链接:
当前的排序顺序 | Last Name 超链接 | Date 超链接 |
---|---|---|
Last Name 升序排列 | descending | ascending |
Last Name 降序排列 | ascending | ascending |
Date 升序排列 | ascending | descending |
Date 降序排列 | ascending | ascending |
该方法使用 LINQ to Entities 指定要作为排序依据的列。 代码在switch 语句之前创建了IQueryable
变量然后在 switch 语句中对其进行修改,并在switch
语句之后调用ToListAsync
方法。 当你创建和修改IQueryable
变量时数据库不会接收到任何查询。 在您调用如ToListAsync
等方法将IQueryable
转换为集合对象之前不会执行查询。 因此,在return View
语句之前此代码只会执行一个查询。
此代码会获得具有大量列的冗长信息。 本系列最后一个教程将演示如何编写代码,使你可以使用字符串将需要OrderBy
的行的名称作为参数传递给方法。
用以下代码替换Views/Students/Index.cshtml,以添加列标题超链接。 高亮代码为已更改的行。
@model IEnumerable<ContosoUniversity.Models.Student>
@{
ViewData["Title"] = "Index";
}
<h2>Indexh2>
<p>
<a asp-action="Create">Create Newa>
p>
<table class="table">
<thead>
<tr>
<th>
<a asp-action="Index" asp-route-sortOrder="@ViewData["NameSortParm"]">@Html.DisplayNameFor(model => model.LastName)a>
th>
<th>
@Html.DisplayNameFor(model => model.FirstMidName)
th>
<th>
<a asp-action="Index" asp-route-sortOrder="@ViewData["DateSortParm"]">@Html.DisplayNameFor(model => model.EnrollmentDate)a>
th>
<th>th>
tr>
thead>
<tbody>
@foreach (var item in Model) {
<tr>
<td>
@Html.DisplayFor(modelItem => item.LastName)
td>
<td>
@Html.DisplayFor(modelItem => item.FirstMidName)
td>
<td>
@Html.DisplayFor(modelItem => item.EnrollmentDate)
td>
<td>
<a asp-action="Edit" asp-route-id="@item.ID">Edita> |
<a asp-action="Details" asp-route-id="@item.ID">Detailsa> |
<a asp-action="Delete" asp-route-id="@item.ID">Deletea>
td>
tr>
}
tbody>
table>
代码中使用了ViewData
元素中的信息来以相应的查询字符串值设置超链接。
运行应用程序中,选择Students卡,然后单击Last Name和Enrollment Date列标题,以验证该排序成功。
向视图添加一个文本框和提交按钮来向索引页添加搜索框,并在Index
方法中做相应更改。 你可以在文本框中输入字符串搜索名字和姓氏字段中的内容。
在StudentsController.cs,将Index
方法替换为以下代码 (突出显示所做的更改)。
public async Task Index(string sortOrder, string searchString)
{
ViewData["NameSortParm"] = String.IsNullOrEmpty(sortOrder) ? "name_desc" : "";
ViewData["DateSortParm"] = sortOrder == "Date" ? "date_desc" : "Date";
ViewData["CurrentFilter"] = searchString;
var students = from s in _context.Students
select s;
if (!String.IsNullOrEmpty(searchString))
{
students = students.Where(s => s.LastName.Contains(searchString)
|| s.FirstMidName.Contains(searchString));
}
switch (sortOrder)
{
case "name_desc":
students = students.OrderByDescending(s => s.LastName);
break;
case "Date":
students = students.OrderBy(s => s.EnrollmentDate);
break;
case "date_desc":
students = students.OrderByDescending(s => s.EnrollmentDate);
break;
default:
students = students.OrderBy(s => s.LastName);
break;
}
return View(await students.AsNoTracking().ToListAsync());
}
在Index
方法中添加searchString
参数。 搜索字符串值来自之后会添加到索引视图中的文本框。 你还向 LINQ 语句 添加了where 子句来选择仅名字或姓氏包含搜索字符串的学生。 添加 where 子句的语句只有在要搜索值的时候才执行。
此处对
IQueryable
对象调用Where
方法,筛选器将在服务器上处理。 在某些场景下你可能会对内存中集合调用作为扩展方法的Where
。 (例如,假设你使用_context.Students
引用,不同于 EFDbSet
,它返回IEnumerable
集合的存储库方法的引用。)结果通常将相同,但在某些情况下可能会不同。例如,默认情况下 .NET Framework 实现的
Contains
方法是对大小写敏感的,但 SQL Server 中由 SQL Server 的排序规则确定。 SQL Server 默认不区分大小写。 您可以调用ToUpper
方法使得其大小写敏感:Where (s = > s.LastName.ToUpper()。Contains(searchString.ToUpper())。 这样做能确保如果将来使用返回IEnumerable
集合的存储库方法而不是IQueryable
对象来修改相关代码,结果还能保持相同。 (当你对IEnumerable
集合调用Contains
方法,你将获取.NET Framework 的实现; 当对IQueryable
对象调用它,则会得到数据库驱动的实现。)但是,此解决方案会对性能产生负面影响。ToUpper
将函数加入到 TSQL SELECT 语句的 WHERE 子句中。 这样做会是的索引优化失去效果。 假设 SQL 大多是是大小写不敏感,在你将数据迁移到大小写敏感的数据存储库之前最好避免ToUpper
代码。
打开Views/Student/Index.cshtml,在table标签之前添加高亮代码以在页面中创建标题,Search文本框,按钮。
<p>
<a asp-action="Create">Create Newa>
p>
<form asp-action="Index" method="get">
<div class="form-actions no-color">
<p>
Find by name: <input type="text" name="SearchString" value="@ViewData["currentFilter"]" />
<input type="submit" value="Search" class="btn btn-default" /> |
<a asp-action="Index">Back to Full Lista>
p>
div>
form>
<table class="table">
此代码通过使用标记帮助器添加搜索文本框和按钮。 默认情况下,
标记帮助器默认使用 post 方法提交数据,这意味着,参数在 HTTP 消息正文中传输表单数据,而不是在 URL 查询字符串上显示并传输。指定使用 HTTP GET 时,表单数据是通过 URL 查询字符串传输,这使得用户能够使用该 URL 来创建书签。 W3C 指南建议当操作未导致更新时使用 GET 方法。
运行应用程序,选择Students选项卡,输入搜索字符串,然后单击搜索以验证筛选是否正常工作。
请注意该 URL 包含搜索字符串。
http://localhost:5813/Students?SearchString=an
如果将此页加入书签,使用书签时你将获得筛选后的列表。 添加method="get"
到form
标签中是导致生成查询字符串的原因。
在此阶段,如果您单击列标题的排序链接你将丢失url上的查询字符串值。 下一部分将修复此问题。
为了向学生索引页添加分页功能,你需要创建PaginatedList
类,该类使用Skip
和Take
语句来对服务器上的数据进行筛选而不是始终检索所有的表行。 接下来你将对Index
方法做更多的修改并将分页按钮添加到Index
视图中。 如下图所示添加了分页按钮的学生索引页。
在项目文件夹中,创建PaginatedList.cs
,然后将模板代码替换为下面的代码。
using System;
using System.Collections.Generic;
using System.Linq;
using System.Threading.Tasks;
using Microsoft.EntityFrameworkCore;
namespace ContosoUniversity
{
public class PaginatedList : List
{
public int PageIndex { get; private set; }
public int TotalPages { get; private set; }
public PaginatedList(List items, int count, int pageIndex, int pageSize)
{
PageIndex = pageIndex;
TotalPages = (int)Math.Ceiling(count / (double)pageSize);
this.AddRange(items);
}
public bool HasPreviousPage
{
get
{
return (PageIndex > 1);
}
}
public bool HasNextPage
{
get
{
return (PageIndex < TotalPages);
}
}
public static async Task> CreateAsync(IQueryable source, int pageIndex, int pageSize)
{
var count = await source.CountAsync();
var items = await source.Skip((pageIndex - 1) * pageSize).Take(pageSize).ToListAsync();
return new PaginatedList(items, count, pageIndex, pageSize);
}
}
}
代码中的CreateAsync
方法获得页面数和当前页码,并对IQueryable
执行 相应的Skip
和Take
语句。 当IQueryable
调用ToListAsync
时,该方法将返回只包含在请求页里的学生列表。 属性HasPreviousPage
和HasNextPage
可用来启用或禁用Previous和Next分页按钮。
由于构造函数里不能运行异步代码,CreateAsync
方法被用作构造函数而只用于创建一个PaginatedList
对象。
在StudentsController.cs中,用以下代码替换Index
方法。
public async Task Index(
string sortOrder,
string currentFilter,
string searchString,
int? page)
{
ViewData["CurrentSort"] = sortOrder;
ViewData["NameSortParm"] = String.IsNullOrEmpty(sortOrder) ? "name_desc" : "";
ViewData["DateSortParm"] = sortOrder == "Date" ? "date_desc" : "Date";
if (searchString != null)
{
page = 1;
}
else
{
searchString = currentFilter;
}
ViewData["CurrentFilter"] = searchString;
var students = from s in _context.Students
select s;
if (!String.IsNullOrEmpty(searchString))
{
students = students.Where(s => s.LastName.Contains(searchString)
|| s.FirstMidName.Contains(searchString));
}
switch (sortOrder)
{
case "name_desc":
students = students.OrderByDescending(s => s.LastName);
break;
case "Date":
students = students.OrderBy(s => s.EnrollmentDate);
break;
case "date_desc":
students = students.OrderByDescending(s => s.EnrollmentDate);
break;
default:
students = students.OrderBy(s => s.LastName);
break;
}
int pageSize = 3;
return View(await PaginatedList.CreateAsync(students.AsNoTracking(), page ?? 1, pageSize));
}
代码中将总页数参数、 当前的排序顺序参数和当前的筛选器参数添加到方法签名中。
public async Task Index(
string sortOrder,
string currentFilter,
string searchString,
int? page)
第一次显示页面,或如果用户未单击分页或排序链接,则所有参数都为null。 如果单击分页链接,页面变量将包含要显示的页码。
名为 CurrentSort 的ViewData
元素提供了当前已排序的试图,因为这必须包含在分页链接中以保持排序顺序在分页时相同。
名为 CurrentFilter 的ViewData
元素提供了当前已筛选的视图。为了在分页过程中维护筛选规则以及在页面重新显示的时候把筛选值恢复到文本框中,该值一定要被包含进分页链接里
如果分页期间更改搜索字符串,显示的页会被重置为 1,因为新的筛选器可能会导致显示不同的数据。 在文本框中输入了值以及按下提交按钮搜索字符串就会改变。 在这种情况下,searchString
参数不为 null。
if (searchString != null)
{
page = 1;
}
else
{
searchString = currentFilter;
}
在Index
方法的结尾,PaginatedList.CreateAsync
方法将学生查询转换为支持分页的集合类型,集合中包含了刚好能放进单页的学生实体。 然后将这个单页大小的学生集合 传递给视图。
return View(await PaginatedList.CreateAsync(students.AsNoTracking(), page ?? 1, pageSize)) ;
PaginatedList.CreateAsync
方法从参数中获取页号。 两个问号表示 null 合并运算符。 Null 合并运算符可以为 null 的类型定义一个默认值; 表达式(page ?? 1)
意味着返回的值如果page
参数为 null 则返回 1,如果指定了一个值则返回指定的值。
在Views/Students/Index.cshtml中,用以下代码替换现有代码。 高亮代码为更改的代码。
@model PaginatedList<ContosoUniversity.Models.Student>
@{
ViewData["Title"] = "Index";
}
<h2>Indexh2>
<p>
<a asp-action="Create">Create Newa>
p>
<form asp-action="Index" method="get">
<div class="form-actions no-color">
<p>
Find by name: <input type="text" name="SearchString" value="@ViewData["currentFilter"]" />
<input type="submit" value="Search" class="btn btn-default" /> |
<a asp-action="Index">Back to Full Lista>
p>
div>
form>
<table class="table">
<thead>
<tr>
<th>
<a asp-action="Index" asp-route-sortOrder="@ViewData["NameSortParm"]" asp-route-currentFilter="@ViewData["CurrentFilter"]">Last Namea>
th>
<th>
First Name
th>
<th>
<a asp-action="Index" asp-route-sortOrder="@ViewData["DateSortParm"]" asp-route-currentFilter="@ViewData["CurrentFilter"]">Enrollment Datea>
th>
<th>th>
tr>
thead>
<tbody>
@foreach (var item in Model)
{
<tr>
<td>
@Html.DisplayFor(modelItem => item.LastName)
td>
<td>
@Html.DisplayFor(modelItem => item.FirstMidName)
td>
<td>
@Html.DisplayFor(modelItem => item.EnrollmentDate)
td>
<td>
<a asp-action="Edit" asp-route-id="@item.ID">Edita> |
<a asp-action="Details" asp-route-id="@item.ID">Detailsa> |
<a asp-action="Delete" asp-route-id="@item.ID">Deletea>
td>
tr>
}
tbody>
table>
@{
var prevDisabled = !Model.HasPreviousPage ? "disabled" : "";
var nextDisabled = !Model.HasNextPage ? "disabled" : "";
}
<a asp-action="Index"
asp-route-sortOrder="@ViewData["CurrentSort"]"
asp-route-page="@(Model.PageIndex - 1)"
asp-route-currentFilter="@ViewData["CurrentFilter"]"
class="btn btn-default @prevDisabled">
Previous
a>
<a asp-action="Index"
asp-route-sortOrder="@ViewData["CurrentSort"]"
asp-route-page="@(Model.PageIndex + 1)"
asp-route-currentFilter="@ViewData["CurrentFilter"]"
class="btn btn-default @nextDisabled">
Next
a>
在页面顶部@model
的语句表示视图现在获取的是PaginatedList
对象而不是List
对象。
列标题链接使用查询字符串将当前的搜索字符串传递到控制器,以便用户可以在筛选结果中进行排序:
<a asp-action="Index" asp-route-sortOrder="@ViewData["DateSortParm"]" asp-route-currentFilter ="@ViewData["CurrentFilter"]">Enrollment Datea>
通过标记帮助程序显示分页按钮:
<a asp-action="Index"
asp-route-sortOrder="@ViewData["CurrentSort"]"
asp-route-page="@(Model.PageIndex - 1)"
asp-route-currentFilter="@ViewData["CurrentFilter"]"
class="btn btn-default @prevDisabled">
Previous
a>
运行应用并转到学生页。
单击以确保分页工作原理的不同的排序顺序中的分页链接。 然后输入搜索字符串,然后重试以验证分页还适用正确使用排序和筛选的分页。
在 Contoso 大学网站About页上,将显示每个课程有多少学生修读。 这要求在分组上再进行分组和简单计算。 要完成此操作,需要执行以下操作:
创建一个视图模型类,该视图类是需要传递到该视图的数据的抽象。
修改对 Home 控制器中的 About 方法。
修改关于视图。
在Model文件夹中创建SchoolViewModels文件夹。
在新的文件夹中,添加EnrollmentDateGroup.cs类文件并将模板代码替换为以下代码:
using System;
using System.ComponentModel.DataAnnotations;
namespace ContosoUniversity.Models.SchoolViewModels
{
public class EnrollmentDateGroup
{
[DataType(DataType.Date)]
public DateTime? EnrollmentDate { get; set; }
public int StudentCount { get; set; }
}
}
在HomeController.cs,在该文件的顶部添加以下 using 语句:
using Microsoft.EntityFrameworkCore;
using ContosoUniversity.Data;
using ContosoUniversity.Models.SchoolViewModels;
在类中,左左大括号后添加的数据库上下文类型的变量,并通过 ASP.NET Core 依赖注入获取上下文的实例:
public class HomeController : Controller
{
private readonly SchoolContext _context;
public HomeController(SchoolContext context)
{
_context = context;
}
将 About
方法的代码替换为以下代码:
public async Task About()
{
IQueryable data =
from student in _context.Students
group student by student.EnrollmentDate into dateGroup
select new EnrollmentDateGroup()
{
EnrollmentDate = dateGroup.Key,
StudentCount = dateGroup.Count()
};
return View(await data.AsNoTracking().ToListAsync());
}
LINQ 语句将学生实体按修读日期分组,计算每个组中的实体数并将结果存储在EnrollmentDateGroup
视图模型对象的集合中。
在 Entity Framework Core 1.0 版本中,整个结果集都会返回到客户端,并在客户端上进行分组。 在某些场景下,这会导致性能问题。 请务必使用用符合生产规模的数据来测试性能,如有必要使用原始 SQL 语句在服务器上进行分组。 有关如何使用原始的 SQL ,请参阅本系列最后一个教程。
将Views/Home/About.cshtml文件替换为以下代码:
@model IEnumerable<ContosoUniversity.Models.SchoolViewModels.EnrollmentDateGroup>
@{
ViewData["Title"] = "Student Body Statistics";
}
<h2>Student Body Statisticsh2>
<table>
<tr>
<th>
Enrollment Date
th>
<th>
Students
th>
tr>
@foreach (var item in Model)
{
<tr>
<td>
@Html.DisplayFor(modelItem => item.EnrollmentDate)
td>
<td>
@item.StudentCount
td>
tr>
}
table>
运行应用并转到关于页面。 表格中显示了每个修读日期的学生计数。
在本教程中,你已了解如何执行排序、 筛选、 分页和分组。 在下一个的教程中,你将了解如何使用迁移来处理数据模型更改。