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

原文 | 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 的更多信息:

原文链接

MongoDB EF Core Provider: What's New?

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

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

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