4

大家好,

今天这是我在我最喜欢的 C# 帮助资源网站上的第一篇文章。我很高兴加入社区。

这是我的问题,我想在使用 .Where 子句过滤双值列表后调用 .Select linq 子句来创建结构列表,但过滤条件似乎没有被评估,或者它们总是返回 true。
请查看我的代码以很好地理解问题:

string[] msgNames = new string[] { "de", "vf", "ze", "ki", "vt", "er" };
double[] prevCounters = new double[] { 154.0, 24588.0, 4547.0, 788.0, 1455.0, 24577.0 };
double[] counters = new double[] { 8548.0, 54854.0, 54854.0, 44.0, 121.0, 48547.0 };
double[] lenValues = new double[] { 1.0, 2.0, 0.0, 4.0, 5.0, 0.0 };

BufferInfo[] positiveLenValues = counters
    .Where((c, it) => lenValues[it] >= 1.0 && prevCounters[it] != c)
    .Select((c, it) =>
    {
        prevCounters[it] = c;

        return new BufferInfo()
        {
            Name = msgNames[it],
            Length = lenValues[it]
        };
    }).ToArray();

也许我误解了 linq 的用法,但最后返回的 BufferInfo[] 包含所有值,而 lenValues 数组中有一些小于 1.0 的值。有关信息,没有对此代码块的并发访问。linq bloc 上的局部变量通常是动态实例化的(由外部程序集重新调整,来自 National Instruments),这里只是为了让您知道。

例如,相同的行为但带有 for 循环也可以:

List<BufferInfo> myInfos = new List<BufferInfo>();

for (int i = 0; i < 6; i++)
{
    if (lenValues[i] >= 1.0 && prevCounters[i] != counters[i])
    {
        oldCounters[i] = counters[i];
        myInfos.Add(new BufferInfo()
                    {
                        Length = lenValues[i],
                        Name = msgNames[i]
                    });
    }
}

BufferInfo[] buffers = myInfos.ToArray();

我不在乎是否使用 linq,我只想了解它为什么会失败。
请解释我为什么会得到这些奇怪的结果。

最好的问候,
Lun@ir。

4

2 回答 2

2

好吧,那是因为你.Select((c, it) =>的索引再次从 0 开始......

因此,如果您的Where子句过滤项目并仅返回 4 个项目,您Select将使用0, 1, 2, 3与您返回的项目的索引不对应的索引Where

于 2013-03-08T16:52:33.847 回答
1

你的代码甚至没有编译,所以我不知道你是如何测试它没有工作的......

您必须先编写Select获取索引,然后将其传递给Where方法:

BufferInfo[] positiveLenValues = counters.Select((c, i) => new { c, i })
                                         .Where(e => lenValues[e.i] >= 1.0 && prevCounters[e.i] != e.c)
                                         .Select(e =>
                                         {
                                             prevCounters[e.i] = e.c;
                                             return new BufferInfo
                                             {
                                                 Name = msgNames[e.i],
                                                 Length = lenValues[e.i]
                                             };
                                         }).ToArray();

该代码返回 4 个值,这是可预测的并且似乎是正确的结果。

于 2013-03-08T16:49:36.630 回答