使用 LINQ 在 DataSet 中使用唯一的列名列表查找 DataTable(Name)

3

我被卷入了一些使用宽松(未经分类)数据集的旧代码中。

我正在尝试编写一个辅助方法来使用一些列的名称查找DataTable.Name.....(因为原始代码对于“有时我们在数据集中有2个Datatables,有时是3个,有时是4个”进行检查)...而且很难知道顺序。 基本上,TSQL Select语句有条件地运行。 (Gaaaaaaaaaaaaaahhh)。

无论如何。 我写了下面的内容,如果我给它两个列名,它会匹配“任何”列名,而不是“所有列名”。

这可能是我的linq技能(再次),也可能是一个简单的修复。

但我已经尝试过了语法糖..下面是我写的其中一件事情,可以编译。

    private static void DataTableFindStuff()
    {

        DataSet ds = new DataSet();

        DataTable dt1 = new DataTable("TableOne");
        dt1.Columns.Add("Table1Column11");
        dt1.Columns.Add("Name");
        dt1.Columns.Add("Age");
        dt1.Columns.Add("Height");

        DataRow row1a = dt1.NewRow();
        row1a["Table1Column11"] = "Table1Column11_ValueA";
        row1a["Name"] = "Table1_Name_NameA";
        row1a["Age"] = "AgeA";
        row1a["Height"] = "HeightA";
        dt1.Rows.Add(row1a);

        DataRow row1b = dt1.NewRow();
        row1b["Table1Column11"] = "Table1Column11_ValueB";
        row1b["Name"] = "Table1_Name_NameB";
        row1b["Age"] = "AgeB";
        row1b["Height"] = "HeightB";
        dt1.Rows.Add(row1b);


        ds.Tables.Add(dt1);

        DataTable dt2 = new DataTable("TableTwo");
        dt2.Columns.Add("Table2Column21");
        dt2.Columns.Add("Name");
        dt2.Columns.Add("BirthCity");
        dt2.Columns.Add("BirthState");


        DataRow row2a = dt2.NewRow();
        row2a["Table2Column21"] = "Table2Column1_ValueG";
        row2a["Name"] = "Table2_Name_NameG";
        row2a["BirthCity"] = "BirthCityA";
        row2a["BirthState"] = "BirthStateA";
        dt2.Rows.Add(row2a);

        DataRow row2b = dt2.NewRow();
        row2b["Table2Column21"] = "Table2Column1_ValueH";
        row2b["Name"] = "Table2_Name_NameH";
        row2b["BirthCity"] = "BirthCityB";
        row2b["BirthState"] = "BirthStateB";
        dt2.Rows.Add(row2b);

        ds.Tables.Add(dt2);

        DataTable dt3 = new DataTable("TableThree");
        dt3.Columns.Add("Table3Column31");
        dt3.Columns.Add("Name");
        dt3.Columns.Add("Price");
        dt3.Columns.Add("QuantityOnHand");


        DataRow row3a = dt3.NewRow();
        row3a["Table3Column31"] = "Table3Column31_ValueM";
        row3a["Name"] = "Table3_Name_Name00M";
        row3a["Price"] = "PriceA";
        row3a["QuantityOnHand"] = "QuantityOnHandA";
        dt3.Rows.Add(row3a);

        DataRow row3b = dt3.NewRow();
        row3b["Table3Column31"] = "Table3Column31_ValueN";
        row3b["Name"] = "Table3_Name_Name00N";
        row3b["Price"] = "PriceB";
        row3b["QuantityOnHand"] = "QuantityOnHandB";
        dt3.Rows.Add(row3b);

        ds.Tables.Add(dt3);


        string foundDataTable1Name = FindDataTableName(ds, new List<string> { "Table1Column11", "Name" });
        /* foundDataTable1Name should be 'TableOne' */

        string foundDataTable2Name = FindDataTableName(ds, new List<string> { "Table2Column21", "Name" });
        /* foundDataTable1Name should be 'TableTwo' */

        string foundDataTable3Name = FindDataTableName(ds, new List<string> { "Table3Column31", "Name" });
        /* foundDataTable1Name should be 'TableThree' */


        string foundDataTableThrowsExceptionName = FindDataTableName(ds, new List<string> { "Name" });
        /* show throw exception as 'Name' is in multiple (distinct) tables */

    }

    public static string FindDataTableName(DataSet ds, List<string> columnNames)
    {
        string returnValue = string.Empty;
        DataTable foundDataTable = FindDataTable(ds, columnNames);
        if (null != foundDataTable)
        {
            returnValue = foundDataTable.TableName;
        }
        return returnValue;
    }

    public static DataTable FindDataTable(DataSet ds, List<string> columnNames)
    {
        DataTable returnItem = null;

        if (null == ds || null == columnNames)
        {
            return null;
        }

        List<DataTable> tables =
                ds.Tables
                .Cast<DataTable>()
                .SelectMany
                    (t => t.Columns.Cast<DataColumn>()
                        .Where(c => columnNames.Contains(c.ColumnName))
                    )
                .Select(c => c.Table).Distinct().ToList();

        if (null != tables)
        {
            if (tables.Count <= 1)
            {
                returnItem = tables.FirstOrDefault();
            }
            else
            {
                throw new IndexOutOfRangeException(string.Format("FindDataTable found more than one matching Table based on the input column names. ({0})", String.Join(", ", columnNames.ToArray())));
            }
        }

        return returnItem;
    }

我也尝试过这个(但是没有成功),总是匹配不到任何结果。
        List<DataTable> tables =
                ds.Tables
                .Cast<DataTable>()
                .Where
                    (t => t.Columns.Cast<DataColumn>()
                        .All(c => columnNames.Contains(c.ColumnName))
                    )
                .Distinct().ToList();

不确定,但如果您将SelectMany替换为简单的Where并将列计数与Where内表中的列进行比较,会发生什么? - Ricardo Souza
1个回答

2

我觉得你是想要确认传递给该方法的columnNames是否包含在TableColumn name集合中。如果是这样,那么下面的代码可以实现。

List<DataTable> tables =
                ds.Tables
                .Cast<DataTable>()
                .Where(dt => !columnNames.Except(dt.Columns.Select(c => c.Name)).Any())
                .ToList();

(以下是提问者追加的内容)

好吧,我得进行一些调整使其编译,但你让我无话可说,谢谢。

最终答案:

        List<DataTable> tables =
                        ds.Tables.Cast<DataTable>()
                        .Where
                            (dt => !columnNames.Except(dt.Columns.Cast<DataColumn>()
                                .Select(c => c.ColumnName))
                                .Any()
                            )
                        .ToList();

好的,请提供需要翻译的内容。
        List<DataTable> tables =
                        ds.Tables.Cast<DataTable>()
                        .Where
                            (dt => !columnNames.Except(dt.Columns.Cast<DataColumn>()
                                .Select(c => c.ColumnName), StringComparer.OrdinalIgnoreCase)
                                .Any()
                            )
                        .ToList();

好的,我找到了另一个定义“ContainsAll”技巧的地方。https://dev59.com/knI_5IYBdhLWcg3wHfOr 这被标记为答案......我一直在苦苦挣扎语法糖...谢谢Michael。 - granadaCoder

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接