C#泛型问题

时间:2009-04-23 06:24:48

标签: c# generics collections

我对泛型有点生疏,尝试做以下事情,但编译器抱怨:

protected List<T> PopulateCollection(DataTable dt) where T: BusinessBase
{
    List<T> lst = new List<T>();
    foreach (DataRow dr in dt.Rows)
    {
        T t = new T(dr);
        lst.Add(t);
    }
    return lst;
}

正如您所看到的,我正在尝试将Table的内容转储到对象中(通过将DataRow传递给构造函数),然后将该对象添加到集合中。它抱怨T不是它所知道的类型或命名空间,并且我不能在非泛型声明中使用它。

这不可能吗?

6 个答案:

答案 0 :(得分:21)

有两个大问题:

  • 您不能指定带参数的构造函数约束
  • 您的方法目前不是通用的 - 它应该是PopulateCollection<T>而不是PopulateCollection

你已经有T : BusinessBase的约束,所以为了解决第一个问题,我建议你在BusinessBase中添加一个抽象(或虚拟)方法:

public abstract void PopulateFrom(DataRow dr);

还向T添加无参数构造函数约束。

您的方法可以成为:

protected List<T> PopulateCollection(DataTable dt)
    where T: BusinessBase, new()
{
    List<T> lst = new List<T>();
    foreach (DataRow dr in dt.Rows)
    {
        T t = new T();
        t.PopulateFrom(dr);
        lst.Add(t);
    }
    return lst;
}

如果您使用的是.NET 3.5,则可以使用DataTableExtensions中的扩展方法稍微简化一下:

protected List<T> PopulateCollection<T>(DataTable dt)
    where T: BusinessBase, new()
{
    return dt.AsEnumerable().Select(dr => 
    { 
        T t = new T();
        t.PopulateFrom(dr);
    }.ToList();
}

或者,您可以将其作为扩展方法本身(再次假设为.NET 3.5)并传入一个函数来返回实例:

static List<T> ToList<T>(this DataTable dt, Func<DataRow dr, T> selector)
    where T: BusinessBase
{
    return dt.AsEnumerable().Select(selector).ToList();
}

然后你的来电者会写:

table.ToList(row => new Whatever(row));

这假设你回到让构造函数采用DataRow。这样做的好处是允许你编写不可变类(以及没有无参数构造函数的类),但这意味着如果没有“工厂”函数,你就无法一般地工作。

答案 1 :(得分:3)

您可以指定允许创建新实例的唯一constraintnew() - 基本上是无参数构造函数。为了避免这种情况:

interface ISupportInitializeFromDataRow
{
    void InitializeFromDataRow(DataRow dataRow);
}

protected List<T> PopulateCollection<T>(DataTable dt) 
    where T : BusinessBase, ISupportInitializeFromDataRow, new()
{
    List<T> lst = new List<T>();
    foreach (DataRow dr in dt.Rows)
    {
        T t = new T();
        t.InitializeFromDataRow(dr);

        lst.Add(t);
    }
    return lst;
}

或者

protected List<T> PopulateCollection<T>(DataTable dt, Func<DataRow, T> builder) 
    where T : BusinessBase
{
    List<T> lst = new List<T>();
    foreach (DataRow dr in dt.Rows)
    {
        T t = builder(dr);        
        lst.Add(t);
    }
    return lst;
}

答案 2 :(得分:2)

您可能需要在T上添加new泛型约束,如下所示:

protected List<T> PopulateCollection<T>(DataTable dt) where T : BusinessBase, new()
...

我无法将DataRow传递给构造函数,但您可以通过将其赋值给BusinessBase

属性来解决这个问题。

答案 3 :(得分:2)

可能的方法是:

protected List<T> PopulateCollection<T>(DataTable dt) where T: BusinessBase, new()
    {
        List<T> lst = new List<T>();
        foreach (DataRow dr in dt.Rows)
        {
            T t = new T();
            t.DataRow = dr;
            lst.Add(t);
        }
        return lst;
    }

答案 4 :(得分:0)

where T: BusinessBase

应该有新的限制()我想添加

答案 5 :(得分:0)

有可能。我的框架中有完全相同的东西。我和你有完全相同的问题,这就是我解决它的方法。从框架中发布相关的片段。如果我记得correclty,最大的问题是需要调用无参数构造函数。

 public class Book<APClass> : Book where APClass : APBase
        private DataTable Table ; // data
        public override IEnumerator GetEnumerator()
        {                        
            for (position = 0;  position < Table.Rows.Count;  position++)           
                 yield return APBase.NewFromRow<APClass>(Table.Rows[position], this.IsOffline);
        }
   ...


  public class APBase ...
  {
    ...
    internal static T NewFromRow<T>(DataRow dr, bool offline) where T : APBase
        {

            Type t = typeof(T);
            ConstructorInfo ci;

            if (!ciDict.ContainsKey(t))
            {
                ci = t.GetConstructor(new Type[1] { typeof(DataRow) });
                ciDict.Add(t, ci);
            }
            else ci = ciDict[t];

            T result = (T)ci.Invoke(new Object[] { dr });

            if (offline)
                result.drCache = dr;    

            return result;
        }

在这种情况下,基类具有静态方法,可以使用接受tablerow的构造函数来实例化其派生类的对象。