繁体   English   中英

Linq返回当前行,其中下一行具有值

[英]Linq return current rows where the next row has value

我需要处理作为字符串的单个Excel列提供给我的发票,我已将其转换为List<string> 简化的示例如下所示。

This is a nothing line
No001 FOO67 368.80
No001 FOO67 17.68
SHORT 12345

在此示例中,我可能需要从以“ No”开头的每一行中提取数据-参考(例如FOO67)和数量(例如368.80)。 但是,如果遇到以“ SHORT”开头的行,则意味着前一行的数量确实是一个调整,并且引用应该是我在“ SHORT”线上找到的任何内容,且金额的符号相反。 在上述情况下,我希望提取的数据如下(第一行是列标题):

Reference    Amount
  FOO67      368.80
  12345      -17.68

我找不到对列表使用linq查询来实现此目的的任何方法。 我认为解决方案可能看起来像这样的模型(由于我已经添加了“ nextline”作为伪代码,因此以下内容将无法解析):

var inv = new List<string> { "This is a nothing line", "No001 FOO67 368.80", "No001 FOO67 17.68", "SHORT 123456" };

var myTable = new DataTable();

myTable.Columns.Add("Id", typeof(int));
myTable.Columns.Add("Ref", typeof(string));
myTable.Columns.Add("Amount", typeof(double));

var foo = from l in inv
            where (l.Substring(0, 2) == "No" && double.TryParse(l.Split(' ')[2], out i))
            select myTable.LoadDataRow(new object[]
                    { inv.IndexOf(l)
                     ,nextline.Contains("SHORT")? nextline.Split(' ')[1] : l.Split(' ')[1]
                     ,nextline.Contains("SHORT")? -1:1 * double.Parse(l.Split(' ')[2].Replace(",", "").Replace("-", ""))
                    }, LoadOption.OverwriteChanges);

有没有一种已知方法可以从查询的下一行获取值并使用它来确定要返回的行? 如果是这样,该怎么做?

您可以使用类似以下的内容。

var inv = new List<string> 
                { 
                 "This is a nothing line", 
                 "No001 FOO67 368.80", 
                 "No001 FOO67 17.68", 
                 "SHORT 123456" 
                };

var filterValidLines = inv.ToLookup(c=>c.StartsWith("No") 
                                              ||c.StartsWith("SHORT"));
var result = filterValidLines[true].Zip(
                            filterValidLines[true].Skip(1),
                            (x,y)=>
{ 
    var subData= x.Split(new []{" "},StringSplitOptions.RemoveEmptyEntries);
    var multiplier = y.StartsWith("SHORT")? -1:1;
    return new Data{Reference= subData[1], Amount = double.Parse(subData[2]) * multiplier};
});

输出量

Reference Amount
FOO67     368.8 
FOO67     -17.68 

快速而肮脏的解决方案:

void Main()
{
    var list = new List<string>
    {
        "This is a nothing line",
        "No001 FOO67 368.80",
        "No001 FOO67 17.68",
        "SHORT 12345"
    };


    var invoices = list
        .Select((l, i) => new InvoiceData
        { 
            Line = l, 
            NextLine = i < list.Count - 1 ? list[i + 1] : string.Empty 
        })
        .Where(x => x.Line.StartsWith("No"))
        .ToList();
}

public class InvoiceData
{
    public string Line { get; set; }
    public string NextLine { get; set; }
    public bool IsAdjustment => NextLine.StartsWith("SHORT");
    public decimal Amount =>
        IsAdjustment 
            ? -decimal.Parse(Line.Split(' ')[2])
            : decimal.Parse(Line.Split(' ')[2]);
    public string Reference =>
        IsAdjustment
            ? NextLine.Split(' ')[1]
            : Line.Split(' ')[1];
}

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM