我可以使用通用约束来启用参数化构造函数吗?

时间:2011-10-14 19:08:22

标签: c# generics

我有这样的功能:

public List<T> SelectAll<T>() where T : DatabaseObject, new()
{
    List<T> retVal = new List<T>();
    String command = "Select * from " + GetType().Name;
    MySqlCommand cmd = DatabaseRunner.GetCommand(command);
    MySqlDataReader reader = cmd.ExecuteReader();
    while (reader.Read())
    {
        T newObj = new T(reader);
    }
    return retVal;
}

您可以看到,当我尝试创建新的T对象时,我将MySqlDataReader对象传递给构造函数。如何修改方法签名以允许此操作?

3 个答案:

答案 0 :(得分:9)

你做不到。 C#仅支持new()约束(您已经使用它),用于将类型参数约束为具有无参数构造函数的类型。在此上下文中不支持具有参数的构造函数。

答案 1 :(得分:8)

简短的回答是你做不到的。解决此问题的一种方法是删除new()约束并使用Activator.CreateInstance()

T value = (T) Activator.CreateInstance(typeof(T), reader)

答案 2 :(得分:7)

您不能使用约束来执行此操作。但是,您可以调用该方法并传入一个函数,以便从T创建MySqlDataReader

public List<T> SelectAll<T>(Func<MySqlDataReader, T> projection)
    where T : DatabaseObject
{
    List<T> retVal = new List<T>();
    String command = "Select * from " + GetType().Name;
    MySqlCommand cmd = DatabaseRunner.GetCommand(command);
    MySqlDataReader reader = cmd.ExecuteReader();
    while (reader.Read())
    {
        retValue.Add(projection(reader));
    }
    return retVal;
}

(您可能希望使用DbDataReader代替MySqlDataReader来避免如此具体。)

另一种方法是给DatabaseObjectInitialize(DbDataReader reader)方法,然后你可以使用:

public List<T> SelectAll<T>() where T : DatabaseObject, new()
{
    List<T> retVal = new List<T>();
    String command = "Select * from " + GetType().Name;
    MySqlCommand cmd = DatabaseRunner.GetCommand(command);
    MySqlDataReader reader = cmd.ExecuteReader();
    while (reader.Read())
    {
        T value = new T();
        value.Initialize(reader);
        retValue.Add(value);
    }
    return retVal;
}

作为一个副问题,你不会处理任何事情 - 你应该使用using语句来自动处理读者和命令。