c # Generics: разные типы возвращаемых данных с использованием Generics - PullRequest
1 голос
/ 08 ноября 2011

У меня есть библиотека доступа к данным, в которую я хотел бы возвращать различные форматы данных (XML, Json, DataTable). Я пытался использовать Generics для достижения этой цели.

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) каждому вызову метода.

Ответы [ 5 ]

0 голосов
/ 08 ноября 2011

Я согласен с 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("");
0 голосов
/ 08 ноября 2011

Что именно вы пытаетесь сделать?Этот код отлично работает:

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
    {
    }
}
0 голосов
/ 08 ноября 2011

Вы можете добавить ограничение new () к определению вашего класса, чтобы ваш класс стал:

public class SomeClass<T> where T : IDBInteractor <T>, new()
{
    public T ExecuteQuery(string myQuery)
    {
       return new T().ExecuteDSQuery(myQuery);
    }
}

Затем вы можете выполнить любую настройку в открытом конструкторе без параметров.

0 голосов
/ 08 ноября 2011

Ваш SomeClass<T> зависит от IDBInteractor<T>.Вы можете добавить ограничение new(), чтобы вы могли создавать экземпляр в SomeClass<T> всякий раз, когда вам это нужно.

Но почему бы не использовать силу внедрения зависимости?Просто примите IDBInteractor<T> в качестве параметра в конструкторе SomeClass<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

0 голосов
/ 08 ноября 2011

если ваш универсальный параметр имеет конструктор без параметров, вы можете создать новый объект T ();

но вы должны определить, что ваш T должен иметь конструктор без параметров используйте общее ограничение new () для принудительного выполнения:

public class SomeClass<T> where T : IDBInteractor <T> , new()
...