Перечислять через подмножество коллекции в C #?
-
23-08-2019 - |
Вопрос
Есть ли хороший способ перечислить только подмножество коллекции в C #?То есть у меня есть коллекция из большого количества объектов (скажем, 1000), но я хотел бы перечислить только элементы 250-340.Есть ли хороший способ получить перечислитель для подмножества коллекции, не используя другую коллекцию?
Редактировать:следовало бы упомянуть, что при этом используется .NET Framework 2.0.
Решение
Попробуйте следующее
var col = GetTheCollection();
var subset = col.Skip(250).Take(90);
Или в более общем плане
public static IEnumerable<T> GetRange(this IEnumerable<T> source, int start, int end) {
// Error checking removed
return source.Skip(start).Take(end - start);
}
Редактировать Решение 2.0
public static IEnumerable<T> GetRange<T>(IEnumerable<T> source, int start, int end ) {
using ( var e = source.GetEnumerator() ){
var i = 0;
while ( i < start && e.MoveNext() ) { i++; }
while ( i < end && e.MoveNext() ) {
yield return e.Current;
i++;
}
}
}
IEnumerable<Foo> col = GetTheCollection();
IEnumerable<Foo> range = GetRange(col, 250, 340);
Другие советы
Мне нравится, чтобы это было просто (если вам не обязательно нужен перечислитель):
for (int i = 249; i < Math.Min(340, list.Count); i++)
{
// do something with list[i]
}
Адаптация оригинального кода Джареда для .Net 2.0:
IEnumerable<T> GetRange(IEnumerable<T> source, int start, int end)
{
int i = 0;
foreach (T item in source)
{
i++;
if (i>end) yield break;
if (i>start) yield return item;
}
}
И использовать его:
foreach (T item in GetRange(MyCollection, 250, 340))
{
// do something
}
Адаптируя код Jarad еще раз, этот метод расширения даст вам подмножество, которое определяется предмет, а не индекс.
//! Get subset of collection between \a start and \a end, inclusive
//! Usage
//! \code
//! using ExtensionMethods;
//! ...
//! var subset = MyList.GetRange(firstItem, secondItem);
//! \endcode
class ExtensionMethods
{
public static IEnumerable<T> GetRange<T>(this IEnumerable<T> source, T start, T end)
{
#if DEBUG
if (source.ToList().IndexOf(start) > source.ToList().IndexOf(end))
throw new ArgumentException("Start must be earlier in the enumerable than end, or both must be the same");
#endif
yield return start;
if (start.Equals(end))
yield break; //start == end, so we are finished here
using (var e = source.GetEnumerator())
{
while (e.MoveNext() && !e.Current.Equals(start)); //skip until start
while (!e.Current.Equals(end) && e.MoveNext()) //return items between start and end
yield return e.Current;
}
}
}
Возможно, вы сможете что-то сделать с Linq.Способ, которым я бы сделал это, - поместить объекты в массив, затем я могу выбрать, какие элементы я хочу обработать, на основе идентификатора массива.
Если вы обнаружите, что вам нужно выполнить изрядное количество нарезок списков и коллекций, возможно, стоит подняться по кривой обучения в библиотека универсальной коллекции C5.