- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
我们在一段使用 LINQ 的代码中遇到了一个轻微的性能问题,它引发了一个关于 LINQ 在查找方面如何工作的问题
我的问题是这样的(请注意,我已经更改了所有代码,所以这是代码的指示性示例,而不是真实场景):
给定
public class Person {
int ID;
string Name;
DateTime Birthday;
int OrganisationID;
}
如果我有一个包含 100k 个 Person 对象的列表,然后是一个日期列表,比如 1000 个,我运行了这段代码:
var personBirthdays = from Person p in personList
where p.OrganisationID = 123
select p.Birthday;
foreach (DateTime d in dateList)
{
if (personBirthdays.Contains(d))
Console.WriteLine(string.Format("Date: {0} has a Birthday", d.ToShortDateString()));
}
生成的代码将是以下内容的迭代:
100k(查找组织ID为123的用户需要执行的循环)
乘以
1000(列表中的日期数量)
乘以
x(组织 ID 为 123 的用户数量,用于检查日期)
这是很多迭代!
如果我将 personBirthdays 的代码更改为:
List<DateTime> personBirthdays =
(from Person p in personList
where p.OrganisationID = 123
select p.Birthday).ToList();
这应该删除 100k 作为倍数,只做一次吗?
所以你会得到 100k + (1000 * x) 而不是 (100k * 1000 * x)。
问题是这看起来太简单了,我确信 LINQ 在某个地方做了一些聪明的事情,这应该意味着这不会发生。
如果没有人回答,我会进行一些测试并报告。
清晰度更新:我们不考虑数据库查找,personList
对象是内存列表对象。这都是 LINQ-to-Objects。
最佳答案
This should remove the 10k as a multiple by, and just do it once?
这意味着不是将 personList
迭代 10 万次,而是为每次迭代执行 where
和 select
操作 您将迭代生成的 List
100k 次,并且 where
和 select
操作只会在基础数据源一次。
The question is that this seems too easy, and I'm sure the LINQ is doing something clever somewhere that should mean that this doesn't happen.
不,你的第一个查询只是一些你不应该使用 LINQ 做的事情,如果你计划多次迭代它们,你应该获取查询的结果并将它们放入数据结构中(这就是你改变了)。
您可以通过使用适当的数据结构进一步改进此查询。在 List
上搜索效率很低,因为它需要进行线性搜索。最好使用 HashSet
来存储查询结果。 HashSet
在平均情况下的搜索速度为 O(1),而 List
的搜索时间为 O(n)。
var dates = new HashSet<DateTime>(from Person p in personList
where p.OrganisationID = 123
select p.Birthday);
foreach (DateTime d in dateList.Where(date => dates.Contains(date)))
{
Console.WriteLine(string.Format("Date: {0} has a Birthday", d.ToShortDateString()));
}
关于c# - 此 LINQ 代码是否对原始数据执行多次查找?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/13823368/
我尝试理解[c代码 -> 汇编]代码 void node::Check( data & _data1, vector& _data2) { -> push ebp -> mov ebp,esp ->
我需要在当前表单(代码)的上下文中运行文本文件中的代码。其中一项要求是让代码创建新控件并将其添加到当前窗体。 例如,在Form1.cs中: using System.Windows.Forms; ..
我有此 C++ 代码并将其转换为 C# (.net Framework 4) 代码。有没有人给我一些关于 malloc、free 和 sprintf 方法的提示? int monate = ee; d
我的网络服务器代码有问题 #include #include #include #include #include #include #include int
给定以下 html 代码,将列表中的第三个元素(即“美丽”一词)以斜体显示的 CSS 代码是什么?当然,我可以给这个元素一个 id 或一个 class,但 html 代码必须保持不变。谢谢
关闭。这个问题不符合Stack Overflow guidelines .它目前不接受答案。 我们不允许提问寻求书籍、工具、软件库等的推荐。您可以编辑问题,以便用事实和引用来回答。 关闭 7 年前。
我试图制作一个宏来避免重复代码和注释。 我试过这个: #define GrowOnPage(any Page, any Component) Component.Width := Page.Surfa
我正在尝试将我的旧 C++ 代码“翻译”成头条新闻所暗示的 C# 代码。问题是我是 C# 中的新手,并不是所有的东西都像 C++ 中那样。在 C++ 中这些解决方案运行良好,但在 C# 中只是不能。我
在 Windows 10 上工作,R 语言的格式化程序似乎没有在 Visual Studio Code 中完成它的工作。我试过R support for Visual Studio Code和 R-T
我正在处理一些报告(计数),我必须获取不同参数的计数。非常简单但乏味。 一个参数的示例查询: qCountsEmployee = ( "select count(*) from %s wher
最近几天我尝试从 d00m 调试网络错误。我开始用尽想法/线索,我希望其他 SO 用户拥有可能有用的宝贵经验。我希望能够提供所有相关信息,但我个人无法控制服务器环境。 整个事情始于用户注意到我们应用程
我有一个 app.js 文件,其中包含如下 dojo amd 模式代码: require(["dojo/dom", ..], function(dom){ dom.byId('someId').i
我对“-gencode”语句中的“code=sm_X”选项有点困惑。 一个例子:NVCC 编译器选项有什么作用 -gencode arch=compute_13,code=sm_13 嵌入库中? 只有
我为我的表格使用 X-editable 框架。 但是我有一些问题。 $(document).ready(function() { $('.access').editable({
我一直在通过本教程学习 flask/python http://blog.miguelgrinberg.com/post/the-flask-mega-tutorial-part-i-hello-wo
我想将 Vim 和 EMACS 用于 CNC、G 代码和 M 代码。 Vim 或 EMACS 是否有任何语法或模式来处理这种类型的代码? 最佳答案 一些快速搜索使我找到了 this vim 和 thi
关闭。这个问题不符合Stack Overflow guidelines .它目前不接受答案。 想改进这个问题?更新问题,使其成为 on-topic对于堆栈溢出。 7年前关闭。 Improve this
这个问题在这里已经有了答案: Enabling markdown highlighting in Vim (5 个回答) 6年前关闭。 当我在 Vim 中编辑包含 Markdown 代码的 READM
我正在 Swift3 iOS 中开发视频应用程序。基本上我必须将视频 Assets 和音频与淡入淡出效果合并为一个并将其保存到 iPhone 画廊。为此,我使用以下方法: private func d
pipeline { agent any stages { stage('Build') { steps { e
我是一名优秀的程序员,十分优秀!