Подпись метода для метода расширения IList <T>.Split () - PullRequest
2 голосов
/ 04 декабря 2009

Я бы хотел написать следующий код:

// contains 500 entries
IList<string> longListOfStrings = ...

// shorterListsOfStrings is now an array of 5 IList<string>,
// with each element containing 100 strings
IList<string>[] shorterListsOfStrings = longListOfStrings.Split(5);

Для этого мне нужно создать общий метод расширения с именем Split, который выглядит примерно так:

public static TList[] Split<TList>(this TList source, int elementCount)
  where TList : IList<>, ICollection<>, IEnumerable<>, IList, ICollection, IEnumerable
{
  return null;
}

Но когда я пытаюсь это скомпилировать, компилятор говорит мне, что IList<>, ICollection<> и IEnumerable<> требуют аргумента типа. Итак, я изменил определение на следующее:

public static TList<TType>[] Split<TList<TType>>(this TList<TType> source, int elementCount)
  where TList : IList<TType>, ICollection<TType>, IEnumerable<TType>, IList, ICollection, IEnumerable
{
  return null;
}

но затем компилятор жалуется, что не может найти тип TList. У меня есть идея, что я слишком усложняю вещи, но я не понимаю, как ... любая помощь ценится!

Ответы [ 2 ]

6 голосов
/ 04 декабря 2009

Да, я думаю, вы слишком усложняете вещи. Попробуйте это:

public static IList<T>[] Split<T>(this IList<T> source, int elementCount)
{
    // What the heck, it's easy to implement...
    IList<T>[] ret = new IList<T>[(source.Count + elementCount - 1) 
                                  / elementCount];
    for (int i = 0; i < ret.Length; i++)
    {
        int start = i * elementCount;
        int size = Math.Min(elementCount, source.Count - i * start);
        T[] tmp = new T[size];
        // Would like CopyTo with a count, but never mind
        for (int j = 0; i < size; j++)
        {
            tmp[j] = source[j + start];
        }
        ret[i] = tmp;
    }
    return ret;
}

В конце концов, вы не собираетесь изменять, какой тип списка вы создаете в методе на основе источника, не так ли? Вы, вероятно, создадите List<T> (или, возможно, T[]), даже если я передам какую-то другую реализацию.

Возможно, вы захотите взглянуть на пакетный метод в MoreLINQ для реализации на основе IEnumerable<T>.

3 голосов
/ 04 декабря 2009

Как насчет этого:

public static IList<TResult>[] Split<TSource, TResult>(
    this IList<TSource> source,      // input IList to split
    Func<TSource, TResult> selector, // projection to apply to each item
    int elementCount                 // number of items per IList
) {
    // do something        
}

И если вам не нужна версия для проецирования каждого элемента:

 public static IList<T>[] Split<T>(
    this IList<T> source, // input IList to split
    int elementCount      // number of items per IList
) {
      return Split<T, T>(source, x => x, elementCount);      
}
...