Вопрос по индексаторам и / или генерикам - PullRequest
3 голосов
/ 27 марта 2009

как можно узнать, реализует ли объект индексатор? Мне нужно поделиться логикой для DataRow и IDataReader, но они не имеют общего интерфейса.

Я также пытался использовать дженерики, но не знаю, какое ограничение я должен наложить на предложение where.

public class Indexer {
    // myObject should be a DataRow or a IDataReader
    private object myObject;
    public object MyObject {
        get { return myObject; }
        set { myObject = value; }
    }
    // won't compile, myObject has no indexer
    public object this[int index] {
        get { return myObject[index]; }
        set { myObject[index] = value; }
    }
    public Indexer(object myObject) {
        this.myObject = myObject;
    }
}

public class Caller {
    void Call() {
        DataRow row = null;
        IDataReader reader = null;
        var ind1 = new Indexer(row);
        var ind2 = new Indexer(reader);
        var val1 = ind1[0];
        var val2 = ind1[0];
    }
}

Ответы [ 3 ]

5 голосов
/ 27 марта 2009

Вам необходимо объявить интерфейс со свойством индексатора, использовать этот интерфейс в качестве ограничения, и класс аргумента типа должен будет реализовать этот интерфейс для удовлетворения ограничения.

Поскольку вы не контролируете классы, которые хотите использовать, это не сработает.

В качестве альтернативы можно заставить класс Indexer принимать операции get / set в качестве отдельных параметров:

public class Indexer {

    private Func<int, object> getter;        
    private Action<int, object> setter;

    public object this[int index] 
    {
        get { return getter(index); }
        set { setter(index, value); }
    }

    public Indexer(Func<int, object> g, Action<int, object> s) 
    {
        getter = g;
        setter = s;
    }
}

public static class IndexerExtensions
{
    public static Indexer ToIndexer(this DataRow row)
    {
        return new Indexer(n => row[n], (n, v) => row[n] = v);
    }

    public static Indexer ToIndexer(this IDataReader row)
    {
        return new Indexer(n => row[n], (n, v) => row[n] = v);
    }
}

Вы можете сделать это:

DataRow row = null;
IDataReader reader = null;
var ind1 = row.ToIndexer();
var ind2 = reader.ToIndexer();
var val1 = ind1[0];
var val2 = ind1[0];
3 голосов
/ 27 марта 2009

Вы можете сделать свой индексатор абстрактным базовым классом с двумя подклассами, один для DataRow и один для IDataReader.

Для упрощения использования могут существовать 2 фабричных метода, таких как: var ind1 = Indexer.CreateFromDataRow(row); var ind2 = Indexer.CreateFromIDataReader(reader);

Они могут создать определенный базовый класс для этого типа с собственной логикой для обработки индексации.

Это позволяет избежать затрат на постоянную проверку типов для каждого вызова get / set (за счет одного виртуального свойства вместо стандартного свойства).

1 голос
/ 27 марта 2009
get { 
    DataRow row = myObject as DataRow;
    if (row != null)
        return row[index];
    IDataReader reader = myObject as IDataReader;
    if (reader != null)
        return reader[index];
}

и использовать ту же логику для множества {}

...