Как получить доступ к последовательным элементам в объекте IQueryable <T>? - PullRequest
3 голосов
/ 02 апреля 2009

Мне нужен доступ к текущему и предыдущему элементу в объекте IQueryable. Если бы у меня был массив int, я бы сделал следующее:

var array = new int[]{0,1,2,3,4};
for(var i = 1; i<array.Length ; i++)
{
    method1(array[i-1], array[i]);
}

Я не знаю, сделать то же самое с IQueryable, так как он не реализует IList.

Ответы [ 5 ]

3 голосов
/ 02 апреля 2009

Использование методов расширения делает это довольно простым.

public static class IEnumerableExtensions
{
  public static IEnumerable<ValueWithPrevious<T>> WithPrevious<T>(this IEnumerable<T> @this)
  {
    using (var e = @this.GetEnumerator())
    {
      if (!e.MoveNext())
        yield break;

      var previous = e.Current;

      while (e.MoveNext())
      {
        yield return new ValueWithPrevious<T>(e.Current, previous);
        previous = e.Current;
      }
    }
  }
}

public struct ValueWithPrevious<T>
{
  public readonly T Value, Previous;

  public ValueWithPrevious(T value, T previous)
  {
    Value = value;
    Previous = previous;
  }
}

Использование:

var array = new int[] { 1, 2, 3, 4, 5 };
foreach (var value in array.WithPrevious())
{
  Console.WriteLine("{0}, {1}", value.Previous, value.Value);
  // Results: 1, 2
  //          2, 3
  //          3, 4
  //          4, 5
}
3 голосов
/ 02 апреля 2009

Вы можете превратить IQueryable <> в список <>, используя ToList <> ().

3 голосов
/ 02 апреля 2009

EDIT

Немного неправильно прочитал вопрос. Этот код даст вам последовательные элементы

public static IEnumerable<Pair<T,T>> GroupIntoConsequetive(this IEnumerable<T> enumerable) {
  using ( var e = enumerable.GetEnumerator() ) {
    if ( !e.MoveNext() ) { 
      yield break;
    }
    var last = e.Current;
    while ( e.MoveNext() ) {
      yield return new Pair<T,T>(last, e.Current);
      last = e.Current;
    }
  }
}

Я не уверен, что есть способ по умолчанию, но написание метода расширения для этого не должно быть трудным. Я предполагаю, что есть простая реализация Pair

public static IEnumerable<Pair<T,T>> Window(this IEnumerable<T> enumerable) {
  using ( var e = enumerable.GetEnumerator() ) {
    while ( e.MoveNext() ) { 
      var first = e.Current;
      if ( !e.MoveNext() ) {
        throw new InvalidOperationException("Need even number");
      }
      var second = e.Current;
      yield return new Pair<T,T>(first,second);
   }
  }
}

С окном вы можете получить желаемое поведение со следующим

var col = GetQueryableItem();
col.Window().Select(pair => method1(pair.First, pair.Second));

Быстрая и грязная пара реализации

public struct Pair<T1,T2> {
  public readonly T1 First;
  public readonly T2 Second;
  public Pair(T1 first, T2 second) {
    First = first;
    Second = second;
  }
}
1 голос
/ 02 апреля 2009

Но он предоставляет методы расширения для создания массива или списка из вашего экземпляра IQueryable<T>, см. ToArray () и ToList () . Затем вы можете пойти и сделать то же самое, что и с массивом в вашем примере.

0 голосов
/ 02 апреля 2009

IQueryable - это IEnumerable. Таким образом, вы можете сделать что-то вроде:

var a = new [] {1, 2, 3, 4}.AsQueryable();

if (a.Count() < 2) {
    return;
}

var prev = a.First();
var isFirst = true;

foreach (var i in a) {
    if (isFirst) {
        isFirst = false;
        continue;
    }

    method1(prev, i);
    prev = i;
}

Или просто преобразовать IQueryable в IList:

var list = a.ToList();
...