我有这样的功能:
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
对象传递给构造函数。如何修改方法签名以允许此操作?
答案 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
来避免如此具体。)
另一种方法是给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;
}
作为一个副问题,你不会处理任何事情 - 你应该使用using
语句来自动处理读者和命令。