将DataTable转换为字典C#

45

我想知道如何将DataTable转换为Dictionary。我尝试了以下代码。

using System.Linq;

internal Dictionary<string,object> GetDict(DataTable dt)
{
    return dt.AsEnumerable()
      .ToDictionary<string, object>(row => row.Field<string>(0),
                                row => row.Field<object>(1));
}

但是我得到了以下错误:

System.Data.EnumerableRowCollection 不包含 'ToDictionary' 的定义,而最佳的扩展方法重载 System.Linq.Parallel.Enumerable.ToDictionary(System.Linq.ParallelQuery, System.Func, System.Collections.Generic.IEqualityComrparer) 具有一些无效的参数。

我该如何解决这个问题?

谢谢


请发布完整的错误信息。 - cadrell0
我添加了完整的错误信息。 - Maximus Decimus
9个回答

101

通用方法ToDictionary有3个参数。您漏掉了一个参数,所以它不知道该怎么做。如果您想指定所有参数,那么应该是<DataRow, string, object>

internal Dictionary<string,object> GetDict(DataTable dt)
{
    return dt.AsEnumerable()
      .ToDictionary<DataRow, string, object>(row => row.Field<string>(0),
                                row => row.Field<object>(1));
}
当然,如果你省略它们,编译器可以推断出类型,因此你不会得到错误。

1
+1,是的,这是主要原因,在ToDictionary<TSource, TKey, TElement>中有更多详细信息。 - Ilya Ivanov
@Ilaria 你必须将其转换为可枚举的方式之一,否则无法使用Linq。另一种方法是 dt.Rows.Cast<DataTable>() - Kilazur
非常感谢... 对我来说起作用了。 - Frijey Labs

20

之前的答案都没有帮到我,所以我做了以下操作:

myList = dt.AsEnumerable()
.ToDictionary<DataRow, string, string>(row => row[0].ToString(),
                                       row => row[1].ToString()); 

而且它运行得非常好!


12

我更喜欢这种方法:

public static List<Dictionary<string, string>> GetDataTableDictionaryList(DataTable dt)
{
    return dt.AsEnumerable().Select(
        row => dt.Columns.Cast<DataColumn>().ToDictionary(
            column => column.ColumnName,
            column => row[column].ToString()
        )).ToList();
}
由于该代码可以通过调用ToString方法处理布尔值或其他数据类型,因此原因就在于此。
请注意,它返回一个字典列表,如果每个行都有键,您可以将其修改为字典的字典。
遍历布尔列可能如下所示:
var list = GetDataTableDictionaryList(dt);

foreach (var row in list)
{
    if (row["Selected"].Equals("true", StringComparison.OrdinalIgnoreCase))
    {
        // do something
    }
}

10

ToDictionary函数的第一个参数需要 IEnumerable<T> 类型,你错误地将其声明为字符串类型,正确的类型应该是 IEnumerable<DataRow>

指定类型时导致程序混淆... 请尝试以下代码...

internal Dictionary<string,object> GetDict(DataTable dt)
{
    return dt.AsEnumerable()
      .ToDictionary(row => row.Field<string>(0),
                                row => row.Field<object>(1));
}

3

给定的解决方案仅假设有2列。 如果您想要多列表示,您需要一个字典列表

class Program
{
    static void Main(string[] args)
    {
        DataTable dt = new DataTable();
        dt.Columns.Add("Column1");
        dt.Columns.Add("Column2");
        dt.Columns.Add("Column3");
        dt.Rows.Add(1, "first", "A");
        dt.Rows.Add(2, "second", "B");

        var dictTable = DataTableToDictionaryList(dt);
        var rowCount = dictTable.Count;
        var colCount = dictTable[0].Count;

        //Linq version
        var dictTableFromLinq = dt.AsEnumerable().Select(
                // ...then iterate through the columns...  
                row => dt.Columns.Cast<DataColumn>().ToDictionary(
                    // ...and find the key value pairs for the dictionary  
                    column => column.ColumnName,    // Key  
                    column => row[column] as string // Value  
                    )
                ).ToList();
    }

    public static List<Dictionary<string, object>> DataTableToDictionaryList(DataTable dt)
    {
        var result = new List<Dictionary<string, object>>();
        //or var result = new List<Dictionary<string, string>>();

        foreach (DataRow row in dt.Rows)
        {
            var dictRow = new Dictionary<string, object>();
            foreach (DataColumn col in dt.Columns)
            {
                dictRow.Add(col.ColumnName, row[col]);
                //or dictRow.Add(col.ColumnName, row[col].ToString());
            }

            result.Add(dictRow);
        }

        return result;
    }
}  

2
我找到了解决方案,但不知道原因。我编辑了我的问题并完善了代码,只是为了让我的操作更加清晰明了,并将其更改为:

    internal Dictionary<string, object> GetDict(DataTable dt)
    {
        Dictionary<String, Object> dic = dt.AsEnumerable().ToDictionary(row => row.Field<String>(0), row => row.Field<Object>(1));
        return dic;
    }

2
今日免费次数已满, 请开通会员/明日再来
            DataTable dt = new DataTable();
            dt.Columns.Add("Column1");
            dt.Columns.Add("Column2");
            dt.Rows.Add(1, "first");
            dt.Rows.Add(2, "second");
            var dictionary = dt.Rows.OfType<DataRow>().ToDictionary(d => d.Field<string>(0), v => v.Field<object>(1));

编译器错误:'DataRow' 不包含 'Field' 的定义,也没有可访问的扩展方法 'Field' 可以接受... - Pawel Cioch
需要在.NET Core中安装System.Data.DataSetExtensions.dll,一切顺利。 - Pawel Cioch

0

Visual Basic
使用 SQL 加载数据表并创建字典

    Dim SalesRep As New System.Collections.Generic.Dictionary(Of String, String)(StringComparer.InvariantCultureIgnoreCase)
    Using tbl As New Data.DataTable("SalesRep")
        SqlCommand.CommandText = "Select Initial,FullName from QB_SalesRep"
        tbl.Load(SqlCommand.ExecuteReader())
        ' --- Option ONE use the array to iterate
        Array.ForEach(tbl.Rows.Cast(Of Data.DataRow).ToArray(),
                      Sub(f)
                          SalesRep.Add(f.ItemAsString("Initial"), f.ItemAsString("FullName"))
                      End Sub)
        ' --- Option TWO use plain linq to create the dictionary
        SalesRep = tbl.Rows _
            .Cast(Of Data.DataRow) _
            .AsEnumerable() _
            .ToDictionary(Function(f) f.Item("Initial").toString(), Function(f) f.Item("FullName").toString)
    End Using

-1

我发现上面的解决方案可行,但它将我的多行表格格式化为仅有2个项目。

为了解决这个问题,我在codegrepper上找到了一些非常有用的东西。

private List<Dictionary<string, object>> DataTableToDictionary(DataTable dt)
{
    var dictionaries = new List<Dictionary<string, object>>();
    foreach (DataRow row in dt.Rows)
    {
        Dictionary<string, object> dictionary = Enumerable.Range(0, dt.Columns.Count).ToDictionary(i => dt.Columns[i].ColumnName, i => row.ItemArray[i]);
        dictionaries.Add(dictionary);
    }

    return dictionaries;
}

参考:codegrepper.com 在C#中将DataTable转换为Dictionary


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