[英]c# Generics: Different Return Types using Generics
我有一个数据访问库,我想返回不同的数据格式(XML,Json,DataTable)。 我试图使用泛型来实现这一目标。
public interface IDBInteractor<T>
{
T ExecuteDSQuery(string myQuery)
}
public class DBInteractorDT : IDBInteractor<DataTable>
{
public DataTable ExecuteDSQuery(string myQuery)
{
return new DataTable();
}
}
public class DBInteractorJson : IDBInteractor<JsonString>
{
public JsonString ExecuteDSQuery(string myQuery)
{
return new JsonString();
}
}
我在调用正确的方法时遇到了麻烦。 理想情况下,我会声明类似
SomeClass<DataTable> dt = new SomeClass<DataTable>();
SomeClass<JsonString> js = new SomeClass<JsonString>();
DataTable myDT = dt.ExecuteDSQuery(myQuery);
JsonString myJson = js.ExecuteDSQuery(myQuery);
我不确定如何声明SomeClass。 我知道我可以做类似的事情
public class SomeClass<T> where T : IDBInteractor <T>
{
public T ExecuteQuery(T dtobject, string myQuery)
{
return dtobject.ExecuteDSQuery(myQuery);
}
}
但是我不想将对象实例(dtobject)传递给每个方法调用。
如果您的通用参数具有无参数构造函数,则可以创建一个新的Object T();
但是您必须定义T必须具有无参数构造函数,并使用通用约束new()强制执行以下操作:
public class SomeClass<T> where T : IDBInteractor <T> , new()
您可以将new()约束添加到类定义中,这样您的类将变为:
public class SomeClass<T> where T : IDBInteractor <T>, new()
{
public T ExecuteQuery(string myQuery)
{
return new T().ExecuteDSQuery(myQuery);
}
}
然后,您可以在公共无参数构造函数中进行任何设置。
您的SomeClass<T>
与IDBInteractor<T>
有依赖关系。 您可以添加new()
约束,以便在需要时在SomeClass<T>
创建一个实例。
但是,为什么不使用依赖注入的功能呢? 只是在SomeClass<T>
的构造函数中接受IDBInteractor<T>
作为参数?
public class SomeClass<T> where T : IDBInteractor <T>
{
private IDBInteractor<T> _interactor;
public SomeClass(IDBInteractor<T> interactor)
{
_interactor = interactor;
}
public T ExecuteQuery(string myQuery)
{
return _interactor.ExecuteDSQuery(myQuery);
}
}
http://www.theserverside.com/news/1321158/A-beginners-guide-to-Dependency-Injection
您到底想做什么? 这段代码可以正常工作:
using System.Data;
namespace ConsoleApplication7
{
internal class Program
{
private static void Main(string[] args)
{
IDBInteractor<DataTable> dt = new SomeClass<DataTable>(new DBInteractorDT());
IDBInteractor<JsonString> js = new SomeClass<JsonString>(new DBInteractorJson());
/*
or you can write
IDBInteractor<DataTable> dt = new DBInteractorDT();
IDBInteractor<JsonString> js = new DBInteractorJson();
*/
DataTable myDT = dt.ExecuteDSQuery("");
JsonString myJson = js.ExecuteDSQuery("");
}
}
public interface IDBInteractor<T>
{
T ExecuteDSQuery(string myQuery);
}
public class DBInteractorDT : IDBInteractor<DataTable>
{
#region IDBInteractor<DataTable> Members
public DataTable ExecuteDSQuery(string myQuery)
{
return new DataTable();
}
#endregion
}
public class DBInteractorJson : IDBInteractor<JsonString>
{
#region IDBInteractor<JsonString> Members
public JsonString ExecuteDSQuery(string myQuery)
{
return new JsonString();
}
#endregion
}
public class SomeClass<T> : IDBInteractor<T>
{
private IDBInteractor<T> interactor;
public SomeClass(IDBInteractor<T> interactor)
{
this.interactor = interactor;
}
public T ExecuteDSQuery(string myQuery)
{
return interactor.ExecuteDSQuery(myQuery);
}
}
public class JsonString
{
}
}
我同意fix_likes_coding。 如果只想创建一个类来执行查询,则可以改用扩展方法。 这可能不太适合解决方案,但是如果您想使事情保持简单,可能会有所帮助。
public static class InteractorUtils
{
public static T ExecuteQuery(this IDBInteractor<T> interactor, string myQuery)
{
return interactor.ExecuteDSQuery(myQuery);
}
}
您可以像这样使用它:
IDBInteractor<DataTable> dt = new DBInteractorDT();
dt.ExecuteQuery("");
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.