public async Task<IActionResult> UpdateCourseCredits(int? multiplier)
if (multiplier != null)
ViewData["RowsAffected"] =
await _context.Database.ExecuteSqlCommandAsync(
"UPDATE Course SET Credits = Credits * {0}",
parameters: multiplier);
return View();
当控制器处理 HttpGet 请求时,ViewData [“RowsAffected”] 中将不返回任何内容,视图显示一个空的文本框和一个提交按钮,如上图所示。
当点击 Update 按钮时,HttpPost 方法被调用, 参数 multiplier 得到文本框中输入的值。 然后代码执行 SQL 更新 Courses 数据表,并将受影响的行数返回到 ViewData 供视图使用。 当视图获得 RowsAffected 值时,显示更新的行数。
在解决方案资源管理器中,右键单击 Views/Courses 文件夹,然后单击 添加>新建项。
在 添加新项 对话框中, 点击左侧面板中 “已安装” 下的 “ASP.NET", 再点击 "MVC View Page (中文:MVC 视图页)",并为新视图命名 UpdateCourseCredits.cshtml 。
在 Views/Courses/UpdateCourseCredits.cshtml 中,用以下代码替换模板代码:
ViewBag.Title = "UpdateCourseCredits";
<h2>Update Course Credits</h2>
@if (ViewData["RowsAffected"] == null)
<form asp-action="UpdateCourseCredits">
<div class="form-actions no-color">
Enter a number to multiply every course's credits by: @Html.TextBox("multiplier")
<input type="submit" value="Update" class="btn btn-default" />
</div>
</form>
@if (ViewData["RowsAffected"] != null)
Number of rows updated: @ViewData["RowsAffected"]
@Html.ActionLink("Back to List", "Index")
</div>
通过点击 Courses 链接,然后在地址栏 URL 后面加入 /UpdateCourseCredits
(例如: http://localhost:5813/Courses/UpdateCourseCredits) 的方法可以运行 UpdateCourseCredits 方法。 在文本框中输入一个数字:
点击 Update 。 你可以看到受影响的行数:
点击 Back to List 查看修改学分后的课程列表。
请注意,在生产代码中,须确保更新总是确保数据是有效的。 此处的简化代码可能导致学分数字大于 5 (通常学分有一个固定范围 0-5 )。 更新查询将可以正常工作,但无效的数据可能将导致系统的其他部分产生意外的结果(因为预先假设学分应该在 0-5 之间)。
有关如何使用原始 SQL 查询的信息, 请参阅 Raw SQL Queries 。
检查发送到数据库的 SQL 语句
有时能够看到发送到数据库的实际 SQL 查询会很有帮助。 EF Core 自动使用 ASP.NET Core 内置的日志记录功能记录查询和更新中包含的 SQL 日志。 在本节中,您将看到一些SQL 日志记录的例子。
打开 StudentsController.cs 并在 Details 方法 if (student == null)
语句上设置一个断点。
在 Debug 模式下运行应用程序, 并转至 学生
的 详细
页面。
转到输出显示调试窗口输出,你可以看到对应的查询语句:
Microsoft.EntityFrameworkCore.Database.Command:Information: Executed DbCommand (56ms) [Parameters=[@__id_0='?'], CommandType='Text', CommandTimeout='30']
SELECT TOP(2) [s].[ID], [s].[Discriminator], [s].[FirstName], [s].[LastName], [s].[EnrollmentDate]
FROM [Person] AS [s]
WHERE ([s].[Discriminator] = N'Student') AND ([s].[ID] = @__id_0)
ORDER BY [s].[ID]
Microsoft.EntityFrameworkCore.Database.Command:Information: Executed DbCommand (122ms) [Parameters=[@__id_0='?'], CommandType='Text', CommandTimeout='30']
SELECT [s.Enrollments].[EnrollmentID], [s.Enrollments].[CourseID], [s.Enrollments].[Grade], [s.Enrollments].[StudentID], [e.Course].[CourseID], [e.Course].[Credits], [e.Course].[DepartmentID], [e.Course].[Title]
FROM [Enrollment] AS [s.Enrollments]
INNER JOIN [Course] AS [e.Course] ON [s.Enrollments].[CourseID] = [e.Course].[CourseID]
INNER JOIN (
SELECT TOP(1) [s0].[ID]
FROM [Person] AS [s0]
WHERE ([s0].[Discriminator] = N'Student') AND ([s0].[ID] = @__id_0)
ORDER BY [s0].[ID]
) AS [t] ON [s.Enrollments].[StudentID] = [t].[ID]
ORDER BY [t].[ID]
你会注意到一些可能让你感到意外的东西:SQL 从 Person 表中选择最多 2 行(TOP(2))。 SingleOrDefaultAsync 方法并不是解析为服务器上的一行。 原因如下:
如果查询可能返回多个行,该方法返回 null。
若要确定查询是否将返回多个行,EF 必须检查是否它返回至少为 2。
请注意,您不必使用调试模式并在断点处停止以在“输出”窗口中获取日志输出。这只是一个方便的方法来停止日志记录在你想看看输出点。 如果你不这样做,日志继续输出,你必须向前滚动滚动条才能找到你感兴趣的部分。仓储和单元工作模式
许多开发人员编写代码来实现仓储和单元工作模式,作为与 Entity Framework 一起工作的包装器。 这些模式旨在创建应用程序中数据访问层和业务逻辑层之间的抽象层。实现这些模式可以帮助您将应用程序与数据存储区中的更改隔离开来,并且可以促进自动化单元测试或测试驱动开发(TDD)。但是,编写额外的代码来实现这些模式并不总使用 EF 开发应用程序的最佳选择,原因如下:
EF 上下文类本身已经将你的代码与特定的数据存储代码分离开了。
EF 上下文类也可以充当 Unit of Work
类来更新数据库。
EF 包含实现 TDD 的特性,无需编写仓储库代码。
有关如何实现仓储和单元工作模式, 请参阅 the Entity Framework 5 Version of this tutorial series. 。
Entity Framework Core 实现了一个内存数据库提供程序。有关更多信息, 请参阅 Testing with InMemory 。自动更改检测
实体框架通过将实体的当前值与原始值进行比较来确定实体如何改变(以及因此需要将哪些更新发送到数据库)。 原始值在实体被查询或附加时被存储。 一些导致自动更改检测的方法如下:
DbContext.SaveChagnes
DbContext.Entry
ChangeTracker.Entries
如果您正在跟踪大量实体,并且您在循环中多次调用其中一个方法,则可以通过使用 ChangeTracker.AutoDetectChangesEnabled 属性临时关闭自动更改检测来显着提高性能。例如:
_context.ChangeTracker.AutoDetectChangesEnabled = false;
Entity Framework Core 源代码及开发计划
Entity Framework Core 的源代码托管在 https://github.com/aspnet/EntityFrameworkCore 。 源码仓库中包含 夜间构建, 问题跟踪, 功能规范, 设计会议记录以及未来发展的路线图。 你可以提交或发行错误,并作出贡献。
尽管源代码是开放的,但 Entity Framework Core 得到了微软的全面支持。 微软 Entity Framework Core 团队保持对贡献的控制,哪些可以被接受,并对所有的代码修改进行测试,以确保每个发行版的质量。
从现有数据库进行逆向工程
要从现有数据库中反向得到数据模型和实体类, 使用 scaffold-dbcontext
命令, 请参阅 getting-started tutorial / 入门教程。
使用 dynamic LINQ 来简化排序选择代码
在 本教程第三章 中, 展示了如何通过在 switch 语句中对列名进行硬编码来编写 LINQ 代码。 此处只有两列可供选择,可以正常工作,但当你有很多列时,代码可能会变得冗长。 为了解决这个问题,可以使用 EF.Property 方法以字符串的形式指定属性的名称。 要尝试这种方法,请使用以下代码替换 StudentsController 中的 Index 方法。
public async Task<IActionResult> Index(
string sortOrder,
string currentFilter,
string searchString,
int? page)
ViewData["CurrentSort"] = sortOrder;
ViewData["NameSortParm"] =
String.IsNullOrEmpty(sortOrder) ? "LastName_desc" : "";
ViewData["DateSortParm"] =
sortOrder == "EnrollmentDate" ? "EnrollmentDate_desc" : "EnrollmentDate";
if (searchString != null)
page = 1;
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));
if (string.IsNullOrEmpty(sortOrder))
sortOrder = "LastName";
bool descending = false;
if (sortOrder.EndsWith("_desc"))
sortOrder = sortOrder.Substring(0, sortOrder.Length - 5);
descending = true;
if (descending)
students = students.OrderByDescending(e => EF.Property<object>(e, sortOrder));
students = students.OrderBy(e => EF.Property<object>(e, sortOrder));
int pageSize = 3;
return View(await PaginatedList<Student>.CreateAsync(students.AsNoTracking(),
page ?? 1, pageSize));
本篇完成了 如何使用 Entity Framewrok Core 创建 ASP.NET MVC Web 应用程序 的教程。
如欲了解有关 EF Core 的更多信息, 请参阅 Entity Framework Core 文档 。 还有一本书可供选择: Entity Framework in Action 。
有关如何部署一个 WEB 应用,请参阅 Host and depoly 。
与 ASP.NET Core MVC 相关的其他主题(如身份验证和授权)的信息,请参阅 ASP.NET Core 文档。
Tom Dykstra 和 Rick Anderson(twitter @RickAndMSFT)编写了这个教程。 Rowan Miller,Diego Vega 和 Entity Framework 团队的其他成员协助代码审查,并帮助调试
在编写教程代码时出现的问题。
ContosoUniversity.dll used by another process
错误信息:
Cannot open '...bin\Debug\netcoreapp1.0\ContosoUniversity.dll' for writing -- 'The process cannot access the file '...\bin\Debug\netcoreapp1.0\ContosoUniversity.dll' because it is being used by another process.
解决方案:
停止 IIS Express 中的站点。 找到 Windows 系统托盘中的 IIS Express ,右键点击图标,选择 Contoso University 站点,单击停止站点。
Migration scaffolded with no code in Up and Down methods
可能的原因:
EF CLI 命令不会自动关闭和保存文件。 如果你在运行 migrations add 命令时有未保存的修改, EF 将找不到你所做的修改。
解决方案:
运行 migrations remove
命令, 保存你的代码修改,然后重新运行 migrations add
命令。
Errors while running database update(运行数据库更新命令时出错)
在已有数据的数据库中进行架构更改,可能会发生其他错误。如果遇到无法解析的迁移错误,则可以更改连接字符串中的数据库名称或删除数据库。 使用新的数据库,不需要迁移任何数据,update-database 命令更可能完成而不会出错。
最简单的方法是在appsettings.json中重命名数据库。 下一次运行数据库更新时,将创建一个新的数据库。
要在 SSOX 中删除一个数据库, 右键点击数据库,单击 删除
, 在 删除数据库
对话框中选项关闭现有连接,然后单击 确定
。
若要使用 CLI 删除数据库,运行 database drop
CLI 命令:
dotnet ef database drop
Error locating SQL Server instance(无法定位SQL Server 实例)
错误信息:
A network-related or instance-specific error occurred while establishing a connection to SQL Server. The server was not found or was not accessible. Verify that the instance name is correct and that SQL Server is configured to allow remote connections. (provider: SQL Network Interfaces, error: 26 - Error Locating Server/Instance Specified)
解决方案:
检查连接字符串。 如果您手动删除了数据库文件,则更改构造字符串中的数据库名称以重建新的数据库。