我有这样的功能:
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对象传递给构造函数.如何修改方法签名以允许此操作?
你不能.C#仅支持new()使用无参数构造函数将类型参数约束为类型的约束(您已使用它).在此上下文中不支持具有参数的构造函数.
简短的回答是你不能.解决此问题的一种方法是删除new()约束并使用Activator.CreateInstance():
T value = (T) Activator.CreateInstance(typeof(T), reader)
Run Code Online (Sandbox Code Playgroud)
你不能用约束来做这件事.但是,您可以调用该方法并传入一个函数来创建一个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语句来自动处理读者和命令.