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

时间:2019-01-16 12:58:33

标签: c# linq

我需要处理作为单个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);

是否存在从查询的下一行获取值并用于确定要返回哪些行的已知方法?如果是这样,该怎么做?

2 个答案:

答案 0 :(得分:1)

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

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 

答案 1 :(得分:1)

一种快速而肮脏的解决方案:

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];
}