一个名为“Item”的列已经属于此DataTable。

3
我将根据您的要求翻译以下内容:

我正在尝试使用扩展方法将List转换为datatable。 实现如下:

扩展方法

public static class list2Dt
    {
        public static DataTable ToDataTable<T>(List<T> items)
        {
            DataTable dataTable = new DataTable(typeof(T).Name);

            //Get all the properties
            PropertyInfo[] Props = typeof(T).GetProperties(BindingFlags.Public | BindingFlags.Instance);
            foreach (PropertyInfo prop in Props)
            {
                //Setting column names as Property names
                dataTable.Columns.Add(prop.Name);
            }
            foreach (T item in items)
            {
                var values = new object[Props.Length];
                for (int i = 0; i < Props.Length; i++)
                {
                    //inserting property values to datatable rows
                    values[i] = Props[i].GetValue(item, null);
                }
                dataTable.Rows.Add(values);
            }
            //put a breakpoint here and check datatable
            return dataTable;
        }
    }

控制器

var noDups = firstTable.AsEnumerable()
                            .GroupBy(d => new
                            {
                                name = d.Field<string>("name"),
                                date = d.Field<string>("date")
                            })
                            .Where(d => d.Count() > 1) 
                            .Select(d => d.First())
                             .ToList();

                        DataTable secondTable = new DataTable();
                        secondTable.Columns.Add("name", typeof(string));
                        secondTable.Columns.Add("date", typeof(string));
                        secondTable.Columns.Add("clockIn", typeof(string));
                        secondTable.Columns.Add("clockOut", typeof(string));

                        secondTable = list2Dt.ToDataTable(noDups);

我遇到了以下错误信息:
An exception of type 'System.Data.DuplicateNameException' occurred in System.Data.dll but was not handled in user code

Additional information: A column named 'Item' already belongs to this DataTable.

上述错误发生在第行:
dataTable.Columns.Add(prop.Name);

有人能够找出问题出在哪里吗。

你的问题是你向数据表中添加了4列,然后调用了 toDataTable 方法尝试添加所有列。 - BugFinder
1
@BugFinder 这不是这种情况下的问题,但由于 OP 在下面调用 ToDataTable,因此这是毫无意义的。 - Jamiec
我错过了 - 很好的观点 - BugFinder
我可能会错,因为术语不是我的强项,但我认为这不是扩展方法 :O - Kritner
1个回答

3
您的ToDataTable方法期望接收一个对象列表,很可能是一组简单的DTO或类似对象。
您传递给它的是一个DataRow实例列表,该类有多个Item属性的重载(请参见此链接),这意味着在尝试构建新的DataTable时,它将会尝试添加多个名称为Item的列,而这在DataTable中是无效的。
解决这个问题的一种方法是将noDups投影到一个新对象中,而不是保留DataRow
public class MyClass
{
    public string Name{get;set;}
    public string Date{get;set;}
}

var noDups = firstTable.AsEnumerable()
                        .GroupBy(d => new
                        {
                            name = d.Field<string>("name"),
                            date = d.Field<string>("date")
                        })
                        .Where(d => d.Count() > 1) 
                        .Select(d => {
                              var first = d.First();
                              return new MyClass()
                                 {
                                     Name = (string)first["name"],
                                     Date = (string)first["date"]
                                 }
                         })

                         .ToList();

干得好。虽然它给了我正确的输出,但我仍然对“该类具有多个属性项重载”的概念感到困惑。这意味着当您尝试构建新的DataTable时,它将尝试添加多个名称为Item的列,这在DataTable中是无效的。 - Jogi
@RehanKhan 我有意将那句话链接到一些文档- 你是否跟随它,阅读并理解了?或者,只需按照您的代码进行步骤,就应该很清楚了。 - Jamiec

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