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

use*_*041 3 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;
}
Run Code Online (Sandbox Code Playgroud)

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

Jon*_*øgh 9

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

  • 我建议改为你想要的是一个接口,它有一个参数,它接受一个`SqlDataReader`并根据需要读取(并初始化)该对象. (2认同)

jru*_*ell 8

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

T value = (T) Activator.CreateInstance(typeof(T), reader)
Run Code Online (Sandbox Code Playgroud)


Jon*_*eet 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;
}
Run Code Online (Sandbox Code Playgroud)

(您可能希望使用DbDataReader而不是MySqlDataReader避免如此具体.)

另一种方法是提供DatabaseObject类似Initialize(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;
}
Run Code Online (Sandbox Code Playgroud)

作为一个副问题,你没有处理任何东西 - 你应该使用using语句来自动处理读者和命令.