c# - 内存数据库提供程序中的实体框架奇怪的行为
问题描述
[HttpGet]
public JsonResult GetTodoItems()
{
return new JsonResult(_context.TodoItems.ToList());
}
[HttpPost]
public JsonResult PostTodoItem(TodoItem item)
{
_context.TodoItems.Add(item);
_context.SaveChanges();
return new JsonResult("Added successfully!");
}
[HttpDelete("{id}")]
public JsonResult DeleteTodoItem(long id)
{
_context.TodoItems.Remove(_context.TodoItems.Find(id));
_context.SaveChanges();
return new JsonResult("Deleted successfully!");
}
另外,这里是 TodoItem 模型:
public class TodoItem
{
public long Id { get; set; }
public string Name { get; set; }
public bool IsComplete { get; set; }
}
出于某种原因,我不明白以下场景的幕后情况:
我使用 API 删除了其中一项。如果我现在将另一个项目添加到列表中,TodoItems 的顺序似乎发生了变化,并且想知道在幕后发生了什么来调用这种行为。
这是一个示例 - 这是起始情况:
[
{ "Id": 1, "Name": "Item1", "IsComplete": false },
{ "Id": 2, "Name": "Item2", "IsComplete": false },
{ "Id": 3, "Name": "Item3", "IsComplete": false }
]
现在我删除 Item2 然后 API 返回的列表是:
[
{ "Id": 1, "Name": "Item1", "IsComplete": false },
{ "Id": 3, "Name": "Item3", "IsComplete": false}
]
然后我添加 Item4 然后列表是:
[
{ "Id": 1, "Name": "Item1", "IsComplete": false },
{ "Id": 4, "Name": "Item4", "IsComplete": false },
{ "Id": 3, "Name": "Item3", "IsComplete": false }
]
如您所见,Item4 被添加到最后删除的项目所在的位置。在此之后,如果我添加后续项目,它会继续在列表末尾添加项目并返回。我猜这可能是因为 EF 的 InMemory Database Provider,因为它主要用于测试目的。但即便如此,我还是想知道这个现象背后的列表是如何运作的。
使用本地监视窗口没有帮助,因为它确实从列表中删除了 Item2 对象,并且显然移动了列表以添加新项目。
编辑:当我在这里谈论列表如何工作时,我的意思是 DbSet 在此添加和删除操作中的工作方式,因为 _context.TodoItems 是一个 DbSet 对象,而 ToList() 只是将其转换为列表。
解决方案
我想知道这个现象背后的列表是如何运作的
每个 EF 提供者都会做这样的事情。此返回的项目顺序:
return new JsonResult(_context.TodoItems.ToList());
是存储提供程序的未记录实现细节,如果您希望以任何特定顺序返回项目,则必须.OrderBy()
在 EF 查询中使用。
如果您对 InMemory 提供程序的实现感到好奇,代码在 GitHub 上,存储是
private readonly Dictionary<TKey, object?[]> _rows;
https://github.com/dotnet/efcore/blob/main/src/EFCore.InMemory/Storage/Internal/InMemoryTable.cs
这导致:
评论
Dictionary<TKey,TValue>.ValueCollection 中值的顺序未指定,但它与 Keys 属性返回的 Dictionary<TKey,TValue>.KeyCollection 中关联键的顺序相同。
推荐阅读
- reactjs - ReactJS:传递道具而不渲染它们
- c# - 图片的本地网址在 Gmail 中不起作用
- css - 在 SAPUI5 中更改图标图像大小和边框
- .net-5 - 多实例问题 - .NET 框架 4.8 WPF 应用程序到 .NET5
- python - Odoo14,外部 API:自定义模型在通过 python 访问时有效,但在通过 PHP 访问时无效
- regex - - 在 powershell 中替换和正则表达式
- python - 我无法在 Windows 上安装 libpostal 库
- c++ - 警告 C4018:“<”:有符号/无符号不匹配?
- m3u8 - 了解 m3u8 文件格式及其工作原理
- jquery - 模态 Ajax 调用不替换数据响应