3

我有一个转换List<T>DataTable使用反射的方法。我想通过传递多个列表来利用该方法创建 DataSet,其中每个列表可能包含不同类型的对象。

下面是给我编译时错误的代码:

The type arguments for method 'ExportToExcel.CreateExcelFile.ListToDataTable<T>(System.Collections.Generic.List<T>)' cannot be inferred from the usage. Try specifying the type arguments explicitly.” - 在行“ var dt = Util.ListToDataTable(item);

public static class Util
    {
        public static DataSet GetDataSet(List<string> title, List<IList> data)
        {
            DataSet ds = new DataSet();
            int idx= 0;
            foreach (var item in data)
            {
                //here I get compile time error "The type arguments for method
                // 'ExportToExcel.CreateExcelFile.ListToDataTable<T>
                // (System.Collections.Generic.List<T>)' cannot be inferred from the usage.
                // Try specifying the type arguments explicitly. "
                var dt = Util.ListToDataTable(item);
                if (title.Count >= idx)
                {
                    dt.TableName = title[idx];
                }
                idx++;
                ds.Tables.Add(dt);
            }
            return ds;
        }

        public static System.Data.DataTable ListToDataTable<T>(List<T> list)
        {
            var dt = new System.Data.DataTable();
            foreach (PropertyInfo info in typeof(T).GetProperties())
            {
                dt.Columns.Add(new DataColumn(info.Name, info.PropertyType));
            }
            foreach (T t in list)
            {
                DataRow row = dt.NewRow();
                foreach (PropertyInfo info in typeof(T).GetProperties())
                {
                    row[info.Name] = info.GetValue(t, null);
                }
                dt.Rows.Add(row);
            }
            return dt;
        }

    }

这是我的调用 testMethod

       [TestMethod]
        public void TestDataSetGeneration_WithMultipleLists()
        {
            IList list = new List<User>();
            list.Add(new User(){FirstName = "Mahesh", LastName = "Chaudhari", IsExternal = true, UpdatedOn = DateTime.Now});
            list.Add(new User(){FirstName = "Mahesh1",LastName = "Chaudhari1",IsExternal = true,UpdatedOn = DateTime.Now});
            list.Add(new User(){FirstName = "Mahesh2",LastName = "Chaudhari2",IsExternal = false,UpdatedOn = DateTime.Now});

            IList hcps = new List<HCPUser>() { new HCPUser(){FirstName = "HCP1",LastName = "HCP1"}};

            var lists = new List<IList>();
            lists.Add(list);
            lists.Add(hcps);

            var titles = new List<String> { "Users", "HCPs"};
            var result = Util.GetDataSet(titles ,lists );
            Assert.IsTrue(result != null);
        }

我认为 Util.ListToDataTable 方法需要特定类型,它只在运行时获得。在这种情况下,我将如何调用此方法?

4

2 回答 2

3

您可以使用MethodInfo.MakeGenericMethod构造一个通用MethodInfo对象并通过反射调用它,但我认为该方法没有理由需要是通用的。ListToDataTable

一种选择是更改List<T>为非通用的东西,例如IEnumerable. 但是,您必须通过从列表中获取第一个元素来找出类型。

另一种选择是 LINQ:

public static System.Data.DataTable ListToDataTable<T>( object[] list )
{
    ....
}

var dt = ListToDataTable( item.Cast<object>().ToArray() );

如果列表为空,这两个选项都将无法找出类型。也许其他人有更整洁的方法。

编辑:我现在看到你的数据从一开始就是非泛型IList的,所以类型基本上已经丢失了。

于 2015-01-15T12:46:01.600 回答
1

似乎代码比需要的要复杂一些。您以不等于泛型类型的泛型类型结束。像上一个答案中所问的那样简单有什么IEnumerable问题?下面的代码通过了您的单元测试。

    public static DataSet GetDataSet(List<string> title, IEnumerable<IEnumerable> data)
    {
        DataSet ds = new DataSet();
        int idx = 0;
        foreach (var item in data)
        {
            //here I get compile time error "The type arguments for method
            // 'ExportToExcel.CreateExcelFile.ListToDataTable<T>
            // (System.Collections.Generic.List<T>)' cannot be inferred from the usage.
            // Try specifying the type arguments explicitly. "
            var dt = Util.ListToDataTable(item);
            if (title.Count >= idx)
            {
                dt.TableName = title[idx];
            }
            idx++;
            ds.Tables.Add(dt);
        }
        return ds;
    }

    public static System.Data.DataTable ListToDataTable(IEnumerable list)
    {
        var dt = new System.Data.DataTable();
        var itm = list.OfType<object>().FirstOrDefault();
        if (itm == null)
            return dt;
        var typeProperties = itm.GetType().GetProperties();

        foreach (PropertyInfo info in typeProperties)
        {
            dt.Columns.Add(new DataColumn(info.Name, info.PropertyType));
        }
        foreach (var t in list)
        {
            DataRow row = dt.NewRow();
            foreach (PropertyInfo info in typeProperties)
            {
                row[info.Name] = info.GetValue(t, null);
            }
            dt.Rows.Add(row);
        }
        return dt;
    }
于 2015-01-15T13:22:40.083 回答