- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
我在家里的电脑上玩 Entity Framework 6,决定尝试插入相当多的行,大约 430k。
我的第一次尝试是这样的,是的,我知道它可以做得更好,但无论如何都是为了研究:
var watch = System.Diagnostics.Stopwatch.StartNew();
foreach (var event in group)
{
db.Events.Add(event);
db.SaveChanges();
}
var dbCount = db.Events.Count(x => x.ImportInformation.FileName == group.Key);
if (dbCount != group.Count())
{
throw new Exception("Mismatch between rows added for file and current number of rows!");
}
watch.Stop();
Console.WriteLine($"Added {dbCount} events to database in {watch.Elapsed.ToString()}");
晚上开始,下类回家后检查。这是结果:
如您所见,在前 4 小时 41 分钟内添加了 64523 个事件,但随后速度变慢了很多,接下来的 66985 个事件耗时 14 小时 51 分钟。我检查了数据库,程序仍在插入事件,但速度极低。然后我决定尝试"new" AddRange DbSet 的方法。
我将我的模型从 IDbSet
切换到 DbSet
并将 foreach
循环替换为:
db.Events.AddRange(group);
db.SaveChanges();
我现在可以在大约 30 秒内添加 60k+ 个事件。它可能不是 SqlBulkCopy很快,但它仍然是一个巨大的进步。为了实现这一目标,幕后发生了什么?我以为我明天要检查 SQL Server Profiler
查询,但最好能解释一下代码中发生的事情。
最佳答案
正如 Jakub 回答的那样,在每个添加的实体之后调用 SaveChanges 没有帮助。但是即使将其移出,您仍然会遇到一些性能问题。这不会解决由 Add 方法引起的性能问题。
使用 Add 方法添加多个实体是一个非常常见的错误。事实上,DetectChanges 方法非常慢。
参见:Entity Framework - Performance Add
It is perhaps not SqlBulkCopy fast, but it is still a huge improvement
有可能获得非常接近 SqlBulkCopy 的性能。
免责声明:我是项目的所有者Entity Framework Extensions
(这个库不是免费的)
此库允许您一次保存多个实体,从而使您的代码更加高效。支持所有批量操作:
例子:
// Easy to use
context.BulkSaveChanges();
// Easy to customize
context.BulkSaveChanges(bulk => bulk.BatchSize = 100);
// Perform Bulk Operations
context.BulkDelete(customers);
context.BulkInsert(customers);
context.BulkUpdate(customers);
// Customize Primary Key
context.BulkMerge(customers, operation => {
operation.ColumnPrimaryKeyExpression =
customer => customer.Code;
});
关于c# - Entity Framework 6 DbSet AddRange vs IDbSet Add - AddRange 怎么能快这么多?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/43643685/
我在家里的电脑上玩 Entity Framework 6,决定尝试插入相当多的行,大约 430k。 我的第一次尝试是这样的,是的,我知道它可以做得更好,但无论如何都是为了研究: var watch =
这个问题在这里已经有了答案: Nested object initializer syntax (1 个回答) 关闭 5 年前。 有一个 C# 功能,我不知道它的术语,因此无法找到关于它的文档。 允
我有一些遗留的 C# 代码,其中包含以下部分: private List mList = new List(); public List getList() { List list = new
通用列表中是否有内置函数可以从特定索引中的另一个列表中添加范围,还是我必须自己编写? 例如: List list1 = new List(); List list2 = new List(); lis
在下面的代码中: var cats = new List() {"cat1", "cat2"}; var dogs = new List() {"dog1", "dog2"}; var animals
具有讽刺意味的是,在将一堆旧 Arraylist 转换为使用通用 List(Of foo) 集合时,为了提高类型安全性,我在 List.AddRange() 中遇到了意外行为方法。 鉴于以下代码,我预
项目:我有一个包含 ComboBox 和 FlowLayoutPanel 的父面板。 FlowLayoutPanel 包含数量可变的子面板(继承自 UserControl 的自定义控件)。每个子面板包
我正在为 ObservableCollection 编写一个扩展方法,并且已了解到 .Add 函数每次调用都会引发 3 个属性更改事件, 所以这样的事情是一个坏主意: public static vo
这个问题在这里已经有了答案: Selection.addRange() is deprecated and will be removed from Chrome (1 个回答) 关闭 5 年前。
基本上我正在寻找这样的东西...... cbo_Genre.Items.AddRange({"Horror", "Comedy"}); 最佳答案 看看这个。 ComboBox1.Items.AddRa
如果我有这两个类: class A {} class B : A {} 我制作了一个 List 但我想通过调用 List.AddRange(List) 添加一个 List 但编译器拒绝: Argume
比如说,我有 3 个列表 List l1 List l1,l2,l3 所有 3 个列表都有很多项目我想将它们全部添加到一个列表中 List finalList finalList.AddRange(l
我坚信 ToolStripItemCollection.AddRange 的实现是一个错误: 我制作了带有两个菜单的 Windows 窗体应用程序,每个菜单包含 2 个项目。 first menu s
此问题中的代码生成以下 JSON。 代码应该根据 .Where(a => a.Values != null) 行排除空的“子”键,但它不起作用。 我可以在哪里放置 where 子句,以便 JSON 不
在HttpWebRequest 的AddRange 方法中,我们可以指定要下载的字节范围。我想知道的是如何指定从某个偏移量到文件末尾的范围。 比如如果我们不知道文件的长度,我们可以在Request H
我已经安装了 Connector/NET 6.5.4。在 VisualStutio 中,IntelliSense 建议使用一种名为 MySqlParameterCollection.AddRange
这似乎是一个简单的问题,但对我来说不是,而且搜索也没有结果。到目前为止,我唯一完成的 .net 编程是使用 Delphi Prism。使用 Prism,我可以执行以下操作: var l := new
“源数组不够长。检查 srcIndex 和长度,以及数组的下限。” 我在创建目标列表(在我添加对象的地方)和源列表(从我在目标列表中添加对象的地方)时没有提到任何大小 这是我正在做的类似代码。 Lis
namespace SubscriptionWebsite.PlaceHolders { public class TreeData { public int ID {
我想做的是 FreezableCollection.AddRange(collectionToAdd) 每次我添加到 FreezableCollection 时都会引发一个事件并发生一些事情。现在我有
我是一名优秀的程序员,十分优秀!