【译】MongoDB EF Core 提供程序:有什么新功能?

news/2025/2/1 20:00:09/文章来源:https://www.cnblogs.com/MingsonZheng/p/18696241

原文 | Rishit, Luce

翻译 | 郑子铭

这是 Rishit Bhatia 和 Luce Carter 的客座文章。Rishit 是 MongoDB 的高级产品经理,专注于 .NET 开发人员体验,在进入产品管理部门之前,他已经使用 C# 工作多年。Luce 是 MongoDB 的开发倡导者、Microsoft MVP,热爱代码、阳光和学习。本博客由 Microsoft .NET 团队针对 EF Core 进行了审阅。

MongoDB 的 EF Core 提供程序于 2024 年 5 月正式发布。自六个月前首次发布此软件包的预览版以来,我们已经取得了长足的进步。我们想分享一些我们一直在研究的有趣功能,如果没有 Microsoft .NET 数据和实体框架团队的支持和合作,这些功能是不可能实现的。

在这篇文章中,我们将使用 MongoDB EF Core 提供程序和 MongoDB Atlas 来展示以下内容:

  • 向实体添加属性并进行更改跟踪
  • 利用出口创建索引
  • 执行复杂查询
  • 事务和乐观并发

与本博客相关的代码可以在 Github 上找到。入门样板代码位于“start”分支中。包含下面提到的所有功能亮点的完整代码位于“main”分支中。

先决条件

我们将使用示例数据集 — 具体来说,本示例中 MongoDB Atlas 可用的 sample_mflix 数据库中的电影集合。要使用示例数据设置 Atlas 集群,您可以按照文档中的步骤操作。我们将创建一个简单的 .NET 控制台应用程序来开始使用 MongoDB EF Core 提供程序。有关如何执行此操作的更多详细信息,您可以查看快速入门指南。

此时,您应该已连接到 Atlas 并能够从快速入门指南中正在读取的电影中输出电影情节。

功能亮点

添加属性和更改跟踪

MongoDB 文档模型的优点之一是它支持灵活的架构。再加上 EF Core 支持 Code First 方法的能力,您可以动态向实体添加属性。为了展示这一点,我们将向我们的模型类添加一个名为 adapted_from_book 的新可空布尔属性。这将使我们的模型类如下所示:

public class Movie
{public ObjectId Id { get; set; }[BsonElement("title")]public string Title { get; set; }[BsonElement("rated")]public string Rated { get; set; }[BsonElement("plot")]public string Plot { get; set; }[BsonElement("adaptedFromBook")]public bool? AdaptedFromBook { get; set; }
}

现在,我们将为找到的电影实体设置这个新添加的属性,并在保存更改后查看 EF Core 的更改跟踪功能。为此,我们将在打印电影情节后添加以下代码行:

movie.AdaptedFromBook = false;
await db.SaveChangesAsync();

在运行程序之前,让我们转到 Atlas 中的集合并找到这部电影,以确保这个新创建的字段 adapted_from_book 不存在于我们的数据库中。为此,只需转到 Atlas Web UI 中的集群并选择浏览集合。

然后,从 sample_mflix 数据库中选择电影集合。在过滤器选项卡中,我们可以使用以下查询找到我们的电影:

{title: "Back to the Future"}

这应该可以找到我们的电影,并且我们可以确认我们想要添加的新字段确实没有被看到。

接下来,让我们在刚刚添加的两行代码中添加一个断点,以确保我们可以在继续操作时实时跟踪更改。选择“开始调试”按钮来运行应用程序。当第一个断点被击中时,我们可以看到本地字段值已被分配。

让我们点击“继续”并检查数据库中的文档。我们可以看到新字段尚未添加。让我们跳过将结束程序的“保存更改”调用。此时,如果我们检查数据库中的文档,我们会注意到新字段已添加,如下所示!

索引管理

MongoDB EF Core 提供程序建立在现有的 .NET/C# 驱动程序之上。此架构的一个优点是,我们可以重用已为 DbContext 创建的 MongoClient,以利用 MongoDB 开发人员数据平台公开的其他功能。这包括但不限于索引管理、Atlas 搜索和矢量搜索等功能。

我们将了解如何在同一个应用程序中使用驱动程序创建新索引。首先,我们将列出集合中的索引,以查看哪些索引已经存在。MongoDB 默认在 _id 字段上创建索引。我们将创建一个辅助函数来打印索引:

var moviesCollection = client.GetDatabase("sample_mflix").GetCollection<Movie>("movies");
Console.WriteLine("Before creating a new Index:");
PrintIndexes();void PrintIndexes()
{var indexes = moviesCollection.Indexes.List();foreach (var index in indexes.ToList()){Console.WriteLine(index);}
}

预期输出如下所示:

{ "v" : 2, "key" : { "_id" : 1 }, "name" : "_id_" }

现在,我们将在我们的集合中的标题和评级字段上创建一个复合索引,并再次打印索引。

var moviesIndex = new CreateIndexModel<Movie>(Builders<Movie>.IndexKeys.Ascending(m => m.Title).Ascending(x => x.Rated));
await moviesCollection.Indexes.CreateOneAsync(moviesIndex);Console.WriteLine("After creating a new Index:");
PrintIndexes();

我们可以看到,一个名为title_1_rated_1的新索引已经创建。

After creating a new Index:
{ "v" : 2, "key" : { "_id" : 1 }, "name" : "_id_" }
{ "v" : 2, "key" : { "title" : 1, "rated" : 1 }, "name" : "title_1_rated_1" }

查询数据

由于 EF Core 已经支持语言集成查询 (LINQ) 语法,因此使用 C# 编写强类型查询变得很容易。根据模型类中可用的字段,我们可以尝试从我们的收藏中查找一些有趣的电影。假设我想查找所有评级为“PG-13”且情节包含单词“shark”的电影,但我希望按标题字段对它们进行排序。我可以使用以下查询轻松完成此操作:

var myMovies = await db.Movies.Where(m => m.Rated == "PG-13" && m.Plot.Contains("shark")).OrderBy(m => m.Title).ToListAsync();foreach (var m in myMovies)
{Console.WriteLine(m.Title);
}

然后,我们可以使用上面的代码打印出查询,并使用 dotnet run 运行程序以查看结果。我们应该能够在控制台中看到我们收藏的 20K+ 部电影中的两部电影名称,如下所示。

Jaws: The Revenge
Shark Night 3D

如果您想查看发送到服务器的查询(在本例中为 MQL),那么您可以在 DbContext 上的 Create 函数中启用日志记录,如下所示:

   public static MflixDbContext Create(IMongoDatabase database) =>new(new DbContextOptionsBuilder<MflixDbContext>().UseMongoDB(database.Client, database.DatabaseNamespace.DatabaseName).LogTo(Console.WriteLine).EnableSensitiveDataLogging().Options);

这样,当我们再次运行程序时,我们就可以看到以下内容作为详细日志的一部分:

Executed MQL query
sample_mflix.movies.aggregate([{ "$match" : { "rated" : "PG-13", "plot" : /shark/s } }, { "$sort" : { "title" : 1 } }])

自动事务和乐观并发

是的,你没看错!MongoDB EF Core 提供程序从其 8.1.0 版本开始支持事务和乐观并发。这意味着默认情况下,SaveChangesSaveChangesAsync 是事务性的。这将使生产级工作负载中的操作在发生任何故障时自动回滚,并确保所有操作都以乐观并发的方式完成。

如果您想关闭事务,您可以在调用任何 SaveChanges 操作之前的初始化阶段进行关闭。

db.Database.AutoTransactionBehavior = AutoTransactionBehavior.Never;

根据您的需求,提供程序支持两种乐观并发方法,即通过并发检查或行版本。您可以在文档中阅读更多相关信息。我们将使用 RowVersion 来演示此用例。这将利用模型类中的 Version 字段,该字段将由 MongoDB EF 提供程序自动更新。要添加版本,我们将以下内容添加到模型类中。

 [Timestamp]public long? Version { get; set; }

首先,让我们创建一个名为 myMovie 的新电影实体,如下所示,并将其添加到 DbSet,然后添加 SaveChangesAsync

Movie myMovie1= new Movie {Title = "The Rise of EF Core 1",Plot = "Entity Framework (EF) Core is a lightweight, extensible, open source and cross-platform version of the popular Entity Framework data access technology.",Rated = "G"
};db.Movies.Add(myMovie1);
await db.SaveChangesAsync();

现在,让我们创建一个类似于上面创建的 DbContext 的新 DbContext。我们可以将数据库创建移到变量中,这样我们就不必再次定义数据库的名称。有了这个新上下文,让我们为电影添加续集并将其添加到 DbSet。我们还将添加第三部分(是的,这是三部曲),但使用与第二部电影实体相同的 ID 到这个新上下文,然后保存我们的更改。

var dbContext2 = MflixDbContext.Create(database);
dbContext2.Database.AutoTransactionBehavior = AutoTransactionBehavior.Never;
var myMovie2 = new Movie { title = "The Rise of EF Core 2" };
dbContext2.Movies.Add(myMovie2);var myMovie3 = new Movie { Id = myMovie2.Id,Title = "The Rise of EF Core 3" };
dbContext2.Movies.Add(myMovie3);
await dbContext2.SaveChangesAsync();

现在支持事务了,对于后两个电影实体的第二组操作不应该通过,因为我们试图用已经存在的 _id 添加它们。我们应该看到一个异常,事务应该只在我们的数据库中看到一部电影。让我们运行一下,看看这是否属实。

我们正确地看到了一个异常,我们可以确认我们只有一部电影(第一部分)插入了数据库。

由于事务已回滚,以下仅显示数据库中的单个文档。

别担心,我们会正确地将我们的三部曲添加到数据库中。让我们删除第三个实体上的 _id 分配,让 MongoDB 自动为我们插入它。

var myMovie3 = new Movie { Title = "The Rise of EF Core 3" };

一旦我们重新运行该程序,我们可以看到所有实体都已添加到数据库中。

摘要

我们能够使用 MongoDB EF Core 提供程序和 MongoDB Atlas 来展示不同的功能,例如动态向实体添加属性、利用 Escape Hatch 创建索引、通过 LINQ 执行复杂查询以及演示新添加的事务和乐观并发支持。

了解更多

要了解有关 EF Core 和 MongoDB 的更多信息:

  • 请参阅 EF Core 文档,了解有关使用 EF Core 访问各种数据库的更多信息。
  • 请参阅 MongoDB 文档,了解有关从任何平台使用 MongoDB 的更多信息。
  • 有关如何开始的更多信息,请参阅 MongoDB EF Core 提供程序文档。
  • 在 Microsoft Youtube 频道上观看有关 EF Core 9:在 .NET 中发展数据访问的演讲。

原文链接

MongoDB EF Core Provider: What’s New?

知识共享许可协议

本作品采用知识共享署名-非商业性使用-相同方式共享 4.0 国际许可协议进行许可。

欢迎转载、使用、重新发布,但务必保留文章署名 郑子铭 (包含链接: http://www.cnblogs.com/MingsonZheng/ ),不得用于商业目的,基于本文修改后的作品务必以相同的许可发布。

如有任何疑问,请与我联系 (MingsonZheng@outlook.com)

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/877630.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

06. 文件权限

一、文件属性Linux 系统是一个典型的多用户系统,不同的用户处于不同的地位,拥有不同的权限。为了保证系统的安全性,Linux 系统对不同的访问用户访问同一个文件(包括目录文件)的权限做了不同的规定。在 Linux 中,我们可以使用 ll 或者 ls -l 命令来显示一个文件的属性以及…

java中的Hashtable

Hashtable Hashtable 是 Java 中的一个古老的集合类,它实现了 Map 接口,基于哈希表存储键值对。 Hashtable 是线程安全的,所有方法都用 synchronized 修饰,因此在多线程环境下可以安全使用。 不过,由于它的性能较差,现代开发中更推荐使用 HashMap Hashtable中的t是小写,…

[HCTF 2018]admin

打开靶机进入登录界面,账号输入admin,密码随便输入一个密码,使用burp抓包后发送到攻击器将密码设为变量,导入字典,开始攻击发现密码"123"的返回长度不同,使用"123"登录,得到flag

DeepSeek LLM

一、背景动机开源社区的关注点:LLaMA 之后,开源社区主要关注训练固定规模的高质量 LLM(如 7B、13B、34B 和 70B),而对 LLM 的缩放定律研究探索较少。缩放定律的重要性:当前开源 LLM 仍处于 AGI 发展的初期阶段,因此研究扩展定律对于未来发展至关重要。缩放结论的分歧:早…

24-25Web开发技术#SpringBoot综合项目实战e1

选用IntelliJ IDEA Community Edition 2024.2 将reggie作为项目(Project)打开 开发环境搭建 系统开发及运行环境 JDK安装尽量遵循原项目的配置,在Oracle官网下载JDK11需注册Oracle账户,较麻烦。 在https://jdk.java.net/archive/找到11 GA (build 11+28)下载Windows版本的压缩…

星球助手发布更新v1.4.0

星球助手发布更新v1.4.0,重要的更新内容有 ​添加了展示评论的功能,可复制评论内容,可展示图片 添加下载评论中的图片的功能 修复了帖子文字展示不全的问题 修复了问答帖子无法搜索的问题 修复了当帖子有多张图片时无法完整显示图片的问题 修复了下载用户信息报错的问题 修复…

CTF-web-POST

1.照样是打开题目链接2.翻译源代码,仍然是向url添加“?what=flag” 3.但是这题需要用hackbar添加4.得到flag

7.机房预约系统

机房预约系统。需求文件目录代码 机房预约系统.cpp #include<iostream> #include "Identity.h" #include<fstream> #include<string> #include "globalFile.h" #include "student.h" #include "teacher.h" #include…

CTF-web-GET

1.打开题目2.翻译原代码,要求添加"?what=flag"3.得到flag:flag{0e8b036fa161804be74c7f33161dc451}

JFrog Artifactory 系列6 --- 其他配置

一、概览 1. 承上启下 JFrog Artifactory 系列 二、其他配置 1. Token过期时间三、参考 https://jfrog.com/help/r/artifactory-intermittent-403-errors-while-working-with-artifactory

新春“码”启 | Cocos 3D 开发微信小游戏(第4天):游戏资源设计和框架核心源代码

新春开发 Cocos 3D 微信小游戏计划的第 4 天,包括游戏资源设计建议,如 UI 背景自适应、碰撞检测组件选用;游戏控制逻辑设计方面,包括组件层级管理及核心代码框架,还有配置中心和消息中心代码,所有的资源与代码均可免费下载……放飞自我的时光总是过得很快,一晃春节就过去…