- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
在这种情况下,我想迭代集合中除最后 2 个元素之外的所有元素。
假设我采用了一种奇怪的方式,例如 x.Reverse().Skip(2).Reverse()
。
每个 LINQ 操作是否会有效地生成一个嵌套迭代器或导致枚举等,还是比这更聪明?在这种情况下会发生什么?
说明:这只是您可能会看到的链式 LINQ 语句的一个示例,其中开发人员喜欢简短而强大的代码,而不太考虑性能 - 也许他们是计算机科学专业的学生,这似乎是“最聪明”的解决方案.我不是在问如何解决这个特定的例子
最佳答案
首先是的,它正在创建一个“迭代器”,并且在您在 foreach
中具体化查询之前实际上不会进行任何迭代或调用 ToList
在上面。当您这样做时,发生的迭代次数基于基础类型。 Reverse
将为您提供的任何源创建一个缓冲区数组并向后迭代它。如果来源是ICollection<T>
然后它将使用它的 CopyTo
填充数组的方法通常会在恒定时间内产生连续数据的单个大容量副本。如果它不是 ICollection<T>
然后它会将源迭代到缓冲区中,然后向后迭代它。考虑到这一点,下面是您的特定查询在迭代时会发生什么。
第一个最后一个Reverse
将开始迭代其源(不是 ICollection<T>
)。
然后 Skip
将开始迭代其来源
然后第一个 Reverse 将执行 CopyTo
如果它的来源是 ICollection<T>
或者它会将源迭代到一个缓冲区数组中,并根据需要调整大小。
然后第一个 Reverse 将向后迭代其缓冲区数组
然后 Skip 将跳过前两个并产生其余的结果
然后最后一个 Reverse 将获取结果并将它们添加到其缓冲区数组并根据需要调整其大小。
最后的 Reverse 将向后迭代缓冲区数组。
因此,如果您要处理 ICollecion<T>
那是一个CopyTo
然后对所有值进行 1 次迭代,然后对除 2 个值之外的所有值进行 1 次迭代。如果它不是 ICollection<T>
这基本上是值的 3 次迭代(实际上最后一次迭代是除了 2 次之外的所有迭代)。无论哪种方式,它还在该过程中使用两个中间数组。
为了证明查询在您具体化之前不会进行迭代,您可以查看此示例
void Main()
{
var query = MyValues().Reverse().Skip(2).Reverse();
Console.WriteLine($"After query before materialization");
var results = query.ToList();
Console.WriteLine(string.Join(",", results));
}
public IEnumerable<int> MyValues()
{
for(int i = 0; i < 10; i ++)
{
Console.WriteLine($"yielding {i}");
yield return i;
}
}
产生输出
After query before materialization
yielding 0
yielding 1
yielding 2
yielding 3
yielding 4
yielding 5
yielding 6
yielding 7
yielding 8
yielding 9
0,1,2,3,4,5,6,7
与另一个例子相比,你有 x.Take(x.Count() - 2)
,这将在您为 Count
实现一次之前迭代源(除非它是 ICollection
或 ICollection<T>
,在这种情况下它将只使用 Count
属性)然后当您实现它时它会再次迭代它。
下面是使用不同代码和结果输出的相同示例。
void Main()
{
var x = MyValues();
var query = x.Take(x.Count() - 2);
Console.WriteLine($"After query before materialization");
var results = query.ToList();
Console.WriteLine(string.Join(",", results));
}
public IEnumerable<int> MyValues()
{
for(int i = 0; i < 10; i ++)
{
Console.WriteLine($"yielding {i}");
yield return i;
}
}
输出
yielding 0
yielding 1
yielding 2
yielding 3
yielding 4
yielding 5
yielding 6
yielding 7
yielding 8
yielding 9
After query before materialization
yielding 0
yielding 1
yielding 2
yielding 3
yielding 4
yielding 5
yielding 6
yielding 7
0,1,2,3,4,5,6,7
所以哪个更好完全取决于出处。对于 ICollection<T>
或 ICollection
Take
和 Count
将是首选(除非源可能在创建查询和实现查询之间发生变化),但如果两者都不是,您可能更喜欢双 Reverse
避免迭代源两次(特别是如果源可以在您创建查询和实际实现它之间发生变化,因为大小也可能发生变化),但这必须根据完成的总迭代次数和内存使用量的增加进行加权。
关于c# - 链式 LINQ 语句的效率如何?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/61548111/
一晃五年没写博客了,依旧再C#上耕耘,依旧没有啥建树,现在也不知道.net上还有多少人再使用,在这里分享一些自己觉得写的还算优雅的代码。 对于自己写着完的代码,我特别喜欢链式(来源于jQuer
我正在构建一个吉他和弦查找应用程序。我使用多维数组来表示指板。数组中的每个元素都由具有字符串属性“Note”的 FretSpace 结构表示。为了初始化指板上的音符属性,我传递了要处理的吉他弦的详细信
我在演示代码中使用 setTimeout 函数模拟了 3 个 ajax 调用。我将从一段运行良好的代码开始:所有调用都是并行进行的,我希望所有调用都能成功,否则会出现错误。 var p1 = func
谁能解释一下? a = [2,3,4] b = [5,6,8,9] print(len(a) > 0) print(len(b) > 0) print((len(a) > 0) & len(b) >
我正在处理具有多个子 JSONObject 的 JSONObject。这是我填写内容的方式: myJson.getJSONObject(CAT_NAME).put(VAR_NAME, var)
想象一下这种情况,我有一个需要检查属性的对象。但是,该对象当前可以具有空值。 如何在一个“if”条件下检查这两个条件? 目前,我必须做这样的事情: if (myObject != null) {
我有一个对象集合,称它们为obj。他们有一个 act() 方法。 act() 方法最终会导致 o 上的 event() observable 调用 onComplete。 链接这些的好方法是什么? 即
假设我有一个列表变量 datalist 存储 10,000 个字符串实体。QTableView 只需要显示其中的一些实体。这就是为什么 QTableView 被指定为 QSortFilterProxy
我正在寻找支持链式 MSI 安装的工具(最好不是 InstallShield,而且最好是便宜/免费的)。我有几个小型安装需要能够单独部署,但也需要作为一个组部署,我不想维护多个安装程序。 看起来我需要
在这种情况下,我想迭代集合中除最后 2 个元素之外的所有元素。 假设我采用了一种奇怪的方式,例如 x.Reverse().Skip(2).Reverse()。 每个 LINQ 操作是否会有效地生成一个
对于javascript来说非常陌生,我有两个html数字选择,包括年份,我想将第二个选择与第一个选择链接起来,这样当我在第一个选择中选择年份时(而第二个选择没有选项)首先),第二个选择应包括从所选数
有人可以向我解释一下为什么以下两个链式函数: // returns zero if okay var resetCounter = function (model) { return new Prom
所以我有 2 个 promise 函数。当第一个函数出现错误时,我希望它显示错误消息。当完成或失败时,我希望他们执行一个finally catch all 函数,但由于某种原因它不起作用。我的代码如下
我有一个函数 const func = () => server.insertPatientSurveyQuestionToDataBase(Store.getPatientID(), SurveyN
(async function() { var a,b; function flush(){ return new Promise(res => {
这个问题已经有答案了: Promise chaining: Use result from previous promise in next then callback [duplicate] (1
这可能不是专业正则表达式理解的问题。唯一重要的是因为我正在运行多个链式替换命令,这些命令会影响文本文件中的某些相同文本。我还想象在替换之前,根据分隔符词(需要多次替换)的使用方式对 txt 文件进行分
我正在尝试构建一组类来定义 OSI 堆栈中协议(protocol)的分层属性...从抽象意义上讲,我只需要从父 python 类继承属性,但我需要能够调用整个类链一次...所以,我正在寻找这样的东西.
我正在努力兑现 promise ,到目前为止我偶然发现了这一点: new Promise((resolve, reject) => { setTimeout(() => { r
我试图理解 promise ,我需要链接它们并装饰来自不同端点的对象宽度数据。 例如: 我的 Node-express 应用程序中有这个 //controller.js export const ge
我是一名优秀的程序员,十分优秀!