{
public static class Enumerable
{
+ enum Fallback {
+ Default,
+ Throw
+ }
+
+ class PredicateOf<T> {
+ public static readonly Func<T, bool> Always = (t) => true;
+ }
+
+ class Function<T> {
+ public static readonly Func<T, T> Identity = (t) => t;
+ }
+
#region Aggregate
public static TSource Aggregate<TSource> (this IEnumerable<TSource> source, Func<TSource, TSource, TSource> func)
if (resultSelector == null)
throw new ArgumentNullException ("resultSelector");
- TAccumulate result = seed;
- foreach (TSource e in source)
+ var result = seed;
+ foreach (var e in source)
result = func (result, e);
return resultSelector (result);
{
Check.SourceAndPredicate (source, predicate);
- foreach (TSource element in source)
+ foreach (var element in source)
if (!predicate (element))
return false;
{
Check.Source (source);
+ var collection = source as ICollection<TSource>;
+ if (collection != null)
+ return collection.Count > 0;
+
using (var enumerator = source.GetEnumerator ())
return enumerator.MoveNext ();
}
public static double Average (this IEnumerable<int> source)
{
- return Average (source, (a, b) => a + b, (a, b) => a / b);
+ return Average<int, long, double> (source, (a, b) => a + b, (a, b) => (double) a / (double) b);
}
public static double Average (this IEnumerable<long> source)
{
- return Average (source, (a, b) => a + b, (a, b) => a / b);
+ return Average<long, long, double> (source, (a, b) => a + b, (a, b) => (double) a / (double) b);
}
public static double Average (this IEnumerable<double> source)
{
- return Average (source, (a, b) => a + b, (a, b) => a / b);
+ return Average<double, double, double> (source, (a, b) => a + b, (a, b) => a / b);
}
public static float Average (this IEnumerable<float> source)
{
- return Average (source, (a, b) => a + b, (a, b) => a / b);
+ return Average<float, double, float> (source, (a, b) => a + b, (a, b) => (float) a / (float) b);
}
public static decimal Average (this IEnumerable<decimal> source)
{
- return Average (source, (a, b) => a + b, (a, b) => a / b);
+ return Average<decimal, decimal, decimal> (source, (a, b) => a + b, (a, b) => a / b);
}
- static TR Average<TA, TR> (this IEnumerable<TA> source, Func<TA, TA, TA> func, Func<TA, int, TR> result)
+ static TResult Average<TElement, TAggregate, TResult> (this IEnumerable<TElement> source,
+ Func<TAggregate, TElement, TAggregate> func, Func<TAggregate, long, TResult> result)
+ where TElement : struct
+ where TAggregate : struct
+ where TResult : struct
{
Check.Source (source);
- TA total = default (TA);
- int counter = 0;
+ var total = default (TAggregate);
+ long counter = 0;
foreach (var element in source) {
total = func (total, element);
++counter;
return result (total, counter);
}
- public static double? Average (this IEnumerable<int?> source)
+ static TResult? AverageNullable<TElement, TAggregate, TResult> (this IEnumerable<TElement?> source,
+ Func<TAggregate, TElement, TAggregate> func, Func<TAggregate, long, TResult> result)
+ where TElement : struct
+ where TAggregate : struct
+ where TResult : struct
{
Check.Source (source);
- bool onlyNull = true;
- long sum = 0;
+ var total = default (TAggregate);
long counter = 0;
- foreach (int? element in source) {
- if (element.HasValue) {
- onlyNull = false;
- sum += element.Value;
- counter++;
- }
+ foreach (var element in source) {
+ if (!element.HasValue)
+ continue;
+
+ total = func (total, element.Value);
+ counter++;
}
- return (onlyNull ? null : (double?) sum / (double?) counter);
+
+ if (counter == 0)
+ return null;
+
+ return new TResult? (result (total, counter));
+ }
+
+ public static double? Average (this IEnumerable<int?> source)
+ {
+ Check.Source (source);
+
+ return source.AverageNullable<int, long, double> ((a, b) => a + b, (a, b) => (double) a / (double) b);
}
public static double? Average (this IEnumerable<long?> source)
{
Check.Source (source);
- bool onlyNull = true;
- long sum = 0;
- long counter = 0;
- foreach (long? element in source) {
- if (element.HasValue) {
- onlyNull = false;
- sum += element.Value;
- counter++;
- }
- }
- return (onlyNull ? null : (double?) sum / (double?) counter);
+ return source.AverageNullable<long, long, double> ((a, b) => a + b, (a, b) => (double) a / b);
}
public static double? Average (this IEnumerable<double?> source)
{
Check.Source (source);
- bool onlyNull = true;
- double sum = 0;
- double counter = 0;
- foreach (double? element in source) {
- if (element.HasValue) {
- onlyNull = false;
- sum += element.Value;
- counter++;
- }
- }
- return (onlyNull ? null : (double?) (sum / counter));
+ return source.AverageNullable<double, double, double> ((a, b) => a + b, (a, b) => a / b);
}
public static decimal? Average (this IEnumerable<decimal?> source)
{
Check.Source (source);
- bool onlyNull = true;
- decimal sum = 0;
- decimal counter = 0;
- foreach (decimal? element in source) {
- if (element.HasValue) {
- onlyNull = false;
- sum += element.Value;
- counter++;
- }
- }
- return (onlyNull ? null : (decimal?) (sum / counter));
+ return source.AverageNullable<decimal, decimal, decimal> ((a, b) => a + b, (a, b) => a / b);
}
public static float? Average (this IEnumerable<float?> source)
{
Check.Source (source);
- float sum = 0;
- float counter = 0;
- foreach (float? element in source) {
- if (element.HasValue) {
- sum += element.Value;
- ++counter;
- }
- }
-
- if (counter == 0)
- return null;
-
- return sum / counter;
+ return source.AverageNullable<float, double, float> ((a, b) => a + b, (a, b) => (float) a / (float) b);
}
public static double Average<TSource> (this IEnumerable<TSource> source, Func<TSource, int> selector)
{
Check.SourceAndSelector (source, selector);
- long sum = 0;
- long counter = 0;
- foreach (TSource item in source) {
- sum += selector (item);
- counter++;
- }
-
- if (counter == 0)
- throw new InvalidOperationException ();
- else
- return (double) sum / (double) counter;
+ return source.Select (selector).Average<int, long, double> ((a, b) => a + b, (a, b) => (double) a / (double) b);
}
public static double? Average<TSource> (this IEnumerable<TSource> source, Func<TSource, int?> selector)
{
Check.SourceAndSelector (source, selector);
- bool onlyNull = true;
- long sum = 0;
- long counter = 0;
- foreach (TSource item in source) {
- int? element = selector (item);
- if (element.HasValue) {
- onlyNull = false;
- sum += element.Value;
- counter++;
- }
- }
- return (onlyNull ? null : (double?) sum / (double?) counter);
+ return source.Select (selector).AverageNullable<int, long, double> ((a, b) => a + b, (a, b) => (double) a / (double) b);
}
public static double Average<TSource> (this IEnumerable<TSource> source, Func<TSource, long> selector)
{
Check.SourceAndSelector (source, selector);
- long sum = 0;
- long counter = 0;
- foreach (TSource item in source) {
- sum += selector (item);
- counter++;
- }
-
- if (counter == 0)
- throw new InvalidOperationException ();
- else
- return (double) sum / (double) counter;
+ return source.Select (selector).Average<long, long, double> ((a, b) => a + b, (a, b) => (double) a / (double) b);
}
public static double? Average<TSource> (this IEnumerable<TSource> source, Func<TSource, long?> selector)
{
Check.SourceAndSelector (source, selector);
- bool onlyNull = true;
- long sum = 0;
- long counter = 0;
- foreach (TSource item in source) {
- long? element = selector (item);
- if (element.HasValue) {
- onlyNull = false;
- sum += element.Value;
- counter++;
- }
- }
- return (onlyNull ? null : (double?) sum / (double?) counter);
+ return source.Select (selector).AverageNullable<long, long, double> ((a, b) => a + b, (a, b) => (double) a / (double) b);
}
public static double Average<TSource> (this IEnumerable<TSource> source, Func<TSource, double> selector)
{
Check.SourceAndSelector (source, selector);
- double sum = 0;
- double counter = 0;
- foreach (TSource item in source) {
- sum += selector (item);
- counter++;
- }
-
- if (counter == 0)
- throw new InvalidOperationException ();
- else
- return sum / counter;
+ return source.Select (selector).Average<double, double, double> ((a, b) => a + b, (a, b) => a / b);
}
public static double? Average<TSource> (this IEnumerable<TSource> source, Func<TSource, double?> selector)
{
Check.SourceAndSelector (source, selector);
- bool onlyNull = true;
- double sum = 0;
- double counter = 0;
- foreach (TSource item in source) {
- double? element = selector (item);
- if (element.HasValue) {
- onlyNull = false;
- sum += element.Value;
- counter++;
- }
- }
- return (onlyNull ? null : (double?) (sum / counter));
+ return source.Select (selector).AverageNullable<double, double, double> ((a, b) => a + b, (a, b) => a / b);
}
public static float Average<TSource> (this IEnumerable<TSource> source, Func<TSource, float> selector)
{
Check.SourceAndSelector (source, selector);
- float sum = 0;
- float counter = 0;
- foreach (TSource item in source) {
- sum += selector (item);
- ++counter;
- }
-
- if (counter == 0)
- throw new InvalidOperationException ();
-
- return sum / counter;
+ return source.Select (selector).Average<float, double, float> ((a, b) => a + b, (a, b) => (float) a / (float) b);
}
public static float? Average<TSource> (this IEnumerable<TSource> source, Func<TSource, float?> selector)
{
Check.SourceAndSelector (source, selector);
- float sum = 0;
- float counter = 0;
- foreach (TSource item in source) {
- float? value = selector (item);
- if (value.HasValue) {
- sum += value.Value;
- ++counter;
- }
- }
-
- if (counter == 0)
- throw new InvalidOperationException ();
-
- return sum / counter;
+ return source.Select (selector).AverageNullable<float, double, float> ((a, b) => a + b, (a, b) => (float) a / (float) b);
}
public static decimal Average<TSource> (this IEnumerable<TSource> source, Func<TSource, decimal> selector)
{
Check.SourceAndSelector (source, selector);
- decimal sum = 0;
- decimal counter = 0;
- foreach (TSource item in source) {
- sum += selector (item);
- counter++;
- }
-
- if (counter == 0)
- throw new InvalidOperationException ();
- else
- return sum / counter;
+ return source.Select (selector).Average<decimal, decimal, decimal> ((a, b) => a + b, (a, b) => a / b);
}
public static decimal? Average<TSource> (this IEnumerable<TSource> source, Func<TSource, decimal?> selector)
{
Check.SourceAndSelector (source, selector);
- bool onlyNull = true;
- decimal sum = 0;
- decimal counter = 0;
- foreach (TSource item in source) {
- decimal? element = selector (item);
- if (element.HasValue) {
- onlyNull = false;
- sum += element.Value;
- counter++;
- }
- }
- return (onlyNull ? null : (decimal?) (sum / counter));
+ return source.Select (selector).AverageNullable<decimal, decimal, decimal> ((a, b) => a + b, (a, b) => a / b);
}
+
#endregion
#region Cast
static IEnumerable<TResult> CreateCastIterator<TResult> (IEnumerable source)
{
- foreach (object element in source)
- yield return (TResult) element;
+ foreach (TResult element in source)
+ yield return element;
}
#endregion
if (comparer == null)
comparer = EqualityComparer<TSource>.Default;
- foreach (TSource e in source) {
- if (comparer.Equals (e, value))
+ foreach (var element in source)
+ if (comparer.Equals (element, value))
return true;
- }
return false;
}
#region ElementAt
+ static TSource ElementAt<TSource> (this IEnumerable<TSource> source, int index, Fallback fallback)
+ {
+ long counter = 0L;
+
+ foreach (var element in source) {
+ if (index == counter++)
+ return element;
+ }
+
+ if (fallback == Fallback.Throw)
+ throw new ArgumentOutOfRangeException ();
+
+ return default (TSource);
+ }
+
public static TSource ElementAt<TSource> (this IEnumerable<TSource> source, int index)
{
Check.Source (source);
+
if (index < 0)
throw new ArgumentOutOfRangeException ();
if (list != null)
return list [index];
- int counter = 0;
- foreach (var element in source) {
- if (counter == index)
- return element;
- counter++;
- }
-
- throw new ArgumentOutOfRangeException ();
+ return source.ElementAt (index, Fallback.Throw);
}
#endregion
public static TSource ElementAtOrDefault<TSource> (this IEnumerable<TSource> source, int index)
{
Check.Source (source);
+
if (index < 0)
return default (TSource);
if (list != null)
return index < list.Count ? list [index] : default (TSource);
- int counter = 0;
- foreach (TSource element in source) {
- if (counter == index)
- return element;
- counter++;
- }
-
- return default (TSource);
+ return source.ElementAt (index, Fallback.Default);
}
#endregion
static IEnumerable<TSource> CreateExceptIterator<TSource> (IEnumerable<TSource> first, IEnumerable<TSource> second, IEqualityComparer<TSource> comparer)
{
- var items = new HashSet<TSource> (Distinct (second));
- foreach (TSource element in first) {
- if (! items.Contains (element, comparer))
+ var items = new HashSet<TSource> (second, comparer);
+ foreach (var element in first) {
+ if (!items.Contains (element, comparer))
yield return element;
}
}
#region First
+ static TSource First<TSource> (this IEnumerable<TSource> source, Func<TSource, bool> predicate, Fallback fallback)
+ {
+ foreach (var element in source)
+ if (predicate (element))
+ return element;
+
+ if (fallback == Fallback.Throw)
+ throw new InvalidOperationException ();
+
+ return default (TSource);
+ }
+
public static TSource First<TSource> (this IEnumerable<TSource> source)
{
Check.Source (source);
- foreach (TSource element in source)
- return element;
+ var list = source as IList<TSource>;
+ if (list != null) {
+ if (list.Count != 0)
+ return list [0];
+
+ throw new InvalidOperationException ();
+ } else {
+ using (var enumerator = source.GetEnumerator ()) {
+ if (enumerator.MoveNext ())
+ return enumerator.Current;
+ }
+ }
throw new InvalidOperationException ();
}
-
public static TSource First<TSource> (this IEnumerable<TSource> source, Func<TSource, bool> predicate)
{
Check.SourceAndPredicate (source, predicate);
- foreach (TSource element in source) {
- if (predicate (element))
- return element;
- }
-
- throw new InvalidOperationException ();
+ return source.First (predicate, Fallback.Throw);
}
#endregion
{
Check.Source (source);
- foreach (TSource element in source)
- return element;
-
- return default (TSource);
+ return source.First (PredicateOf<TSource>.Always, Fallback.Default);
}
-
public static TSource FirstOrDefault<TSource> (this IEnumerable<TSource> source, Func<TSource, bool> predicate)
{
Check.SourceAndPredicate (source, predicate);
- foreach (TSource element in source) {
- if (predicate (element))
- return element;
- }
-
- return default (TSource);
+ return source.First (predicate, Fallback.Default);
}
#endregion
return null;
}
-
public static IEnumerable<IGrouping<TKey, TSource>> GroupBy<TSource, TKey> (this IEnumerable<TSource> source,
Func<TSource, TKey> keySelector)
{
{
Check.SourceAndKeySelector (source, keySelector);
+ return CreateGroupByIterator (source, keySelector, comparer);
+ }
+
+ static IEnumerable<IGrouping<TKey, TSource>> CreateGroupByIterator<TSource, TKey> (this IEnumerable<TSource> source,
+ Func<TSource, TKey> keySelector, IEqualityComparer<TKey> comparer)
+ {
Dictionary<TKey, List<TSource>> groups = new Dictionary<TKey, List<TSource>> ();
List<TSource> nullList = new List<TSource> ();
int counter = 0;
}
}
-
public static IEnumerable<IGrouping<TKey, TElement>> GroupBy<TSource, TKey, TElement> (this IEnumerable<TSource> source,
Func<TSource, TKey> keySelector, Func<TSource, TElement> elementSelector)
{
return GroupBy (source, keySelector, elementSelector, resultSelector, null);
}
- [MonoTODO]
public static IEnumerable<TResult> GroupBy<TSource, TKey, TElement, TResult> (this IEnumerable<TSource> source,
Func<TSource, TKey> keySelector, Func<TSource, TElement> elementSelector,
Func<TKey, IEnumerable<TElement>, TResult> resultSelector,
IEqualityComparer<TKey> comparer)
{
- throw new NotImplementedException ();
+ IEnumerable<IGrouping<TKey, TElement>> groups = GroupBy<TSource, TKey, TElement> (
+ source, keySelector, elementSelector, comparer);
+
+ foreach (IGrouping<TKey, TElement> group in groups)
+ yield return resultSelector (group.Key, group);
}
public static IEnumerable<TResult> GroupBy<TSource, TKey, TResult> (this IEnumerable<TSource> source,
return GroupBy (source, keySelector, resultSelector, null);
}
- [MonoTODO]
public static IEnumerable<TResult> GroupBy<TSource, TKey, TResult> (this IEnumerable<TSource> source,
Func<TSource, TKey> keySelector,
Func<TKey, IEnumerable<TSource>, TResult> resultSelector,
IEqualityComparer<TKey> comparer)
{
- throw new NotImplementedException ();
+ IEnumerable<IGrouping<TKey,TSource>> groups = GroupBy<TSource, TKey> (source, keySelector, comparer);
+
+ foreach (IGrouping<TKey, TSource> group in groups)
+ yield return resultSelector (group.Key, group);
}
#endregion
if (comparer == null)
comparer = EqualityComparer<TKey>.Default;
+ return CreateGroupJoinIterator (outer, inner, outerKeySelector, innerKeySelector, resultSelector, comparer);
+ }
+
+ static IEnumerable<TResult> CreateGroupJoinIterator<TOuter, TInner, TKey, TResult> (this IEnumerable<TOuter> outer,
+ IEnumerable<TInner> inner, Func<TOuter, TKey> outerKeySelector,
+ Func<TInner, TKey> innerKeySelector, Func<TOuter, IEnumerable<TInner>, TResult> resultSelector,
+ IEqualityComparer<TKey> comparer)
+ {
ILookup<TKey, TInner> innerKeys = ToLookup<TInner, TKey> (inner, innerKeySelector, comparer);
/*Dictionary<K, List<U>> innerKeys = new Dictionary<K, List<U>> ();
foreach (U element in inner)
{
var items = new HashSet<TSource> (second, comparer);
foreach (TSource element in first) {
- if (items.Contains (element))
+ if (items.Remove (element))
yield return element;
}
}
if (comparer == null)
comparer = EqualityComparer<TKey>.Default;
+ return CreateJoinIterator (outer, inner, outerKeySelector, innerKeySelector, resultSelector, comparer);
+ }
+
+ static IEnumerable<TResult> CreateJoinIterator<TOuter, TInner, TKey, TResult> (this IEnumerable<TOuter> outer,
+ IEnumerable<TInner> inner, Func<TOuter, TKey> outerKeySelector,
+ Func<TInner, TKey> innerKeySelector, Func<TOuter, TInner, TResult> resultSelector, IEqualityComparer<TKey> comparer)
+ {
ILookup<TKey, TInner> innerKeys = ToLookup<TInner, TKey> (inner, innerKeySelector, comparer);
/*Dictionary<K, List<U>> innerKeys = new Dictionary<K, List<U>> ();
foreach (U element in inner)
IEnumerable<TInner> inner, Func<TOuter, TKey> outerKeySelector,
Func<TInner, TKey> innerKeySelector, Func<TOuter, TInner, TResult> resultSelector)
{
- return Join<TOuter, TInner, TKey, TResult> (outer, inner, outerKeySelector, innerKeySelector, resultSelector, null);
+ return outer.Join (inner, outerKeySelector, innerKeySelector, resultSelector, null);
}
- # endregion
+ #endregion
#region Last
- public static TSource Last<TSource> (this IEnumerable<TSource> source)
+ static TSource Last<TSource> (this IEnumerable<TSource> source, Func<TSource, bool> predicate, Fallback fallback)
{
- Check.Source (source);
+ var empty = true;
+ var item = default (TSource);
- bool noElements = true;
- TSource lastElement = default (TSource);
- foreach (TSource element in source) {
- if (noElements) noElements = false;
- lastElement = element;
+ foreach (var element in source) {
+ if (!predicate (element))
+ continue;
+
+ item = element;
+ empty = false;
}
- if (!noElements)
- return lastElement;
- else
+ if (!empty)
+ return item;
+
+ if (fallback == Fallback.Throw)
throw new InvalidOperationException ();
+
+ return item;
}
- public static TSource Last<TSource> (this IEnumerable<TSource> source,
- Func<TSource, bool> predicate)
+ public static TSource Last<TSource> (this IEnumerable<TSource> source)
{
- Check.SourceAndPredicate (source, predicate);
-
- bool noElements = true;
- TSource lastElement = default (TSource);
- foreach (TSource element in source) {
- if (predicate (element)) {
- if (noElements) noElements = false;
- lastElement = element;
- }
- }
+ Check.Source (source);
- if (!noElements)
- return lastElement;
- else
+ var collection = source as ICollection<TSource>;
+ if (collection != null && collection.Count == 0)
throw new InvalidOperationException ();
+
+ var list = source as IList<TSource>;
+ if (list != null)
+ return list [list.Count - 1];
+
+ return source.Last (PredicateOf<TSource>.Always, Fallback.Throw);
+ }
+
+ public static TSource Last<TSource> (this IEnumerable<TSource> source, Func<TSource, bool> predicate)
+ {
+ Check.SourceAndPredicate (source, predicate);
+
+ return source.Last (predicate, Fallback.Throw);
}
#endregion
if (list != null)
return list.Count > 0 ? list [list.Count - 1] : default (TSource);
- TSource lastElement = default (TSource);
- foreach (TSource element in source)
- lastElement = element;
-
- return lastElement;
+ return source.Last (PredicateOf<TSource>.Always, Fallback.Default);
}
- public static TSource LastOrDefault<TSource> (this IEnumerable<TSource> source,
- Func<TSource, bool> predicate)
+ public static TSource LastOrDefault<TSource> (this IEnumerable<TSource> source, Func<TSource, bool> predicate)
{
Check.SourceAndPredicate (source, predicate);
- TSource lastElement = default (TSource);
- foreach (TSource element in source) {
- if (predicate (element))
- lastElement = element;
- }
-
- return lastElement;
+ return source.Last (predicate, Fallback.Default);
}
#endregion
#region LongCount
+
public static long LongCount<TSource> (this IEnumerable<TSource> source)
{
Check.Source (source);
+#if !NET_2_1
+ var array = source as TSource [];
+ if (array != null)
+ return array.LongLength;
+#endif
+
long counter = 0;
using (var enumerator = source.GetEnumerator ())
while (enumerator.MoveNext ())
public static int Max (this IEnumerable<int> source)
{
- return Iterate (source, int.MinValue, (a, b) => a > b);
+ Check.Source (source);
+
+ return Iterate (source, int.MinValue, (a, b) => Math.Max (a, b));
}
public static long Max (this IEnumerable<long> source)
{
- return Iterate (source, long.MinValue, (a, b) => a > b);
+ Check.Source (source);
+
+ return Iterate (source, long.MinValue, (a, b) => Math.Max (a, b));
}
public static double Max (this IEnumerable<double> source)
{
- return Iterate (source, double.MinValue, (a, b) => a > b);
+ Check.Source (source);
+
+ return Iterate (source, double.MinValue, (a, b) => Math.Max (a, b));
}
public static float Max (this IEnumerable<float> source)
{
- return Iterate (source, float.MinValue, (a, b) => a > b);
- }
+ Check.Source (source);
- public static decimal Max (this IEnumerable<decimal> source)
- {
- return Iterate (source, decimal.MinValue, (a, b) => a > b);
+ return Iterate (source, float.MinValue, (a, b) => Math.Max (a, b));
}
- static T Iterate<T> (IEnumerable<T> source, T initValue, Func<T, T, bool> selector)
+ public static decimal Max (this IEnumerable<decimal> source)
{
- Check.SourceAndSelector (source, selector);
-
- int counter = 0;
- foreach (var element in source) {
- if (selector (element, initValue))
- initValue = element;
- ++counter;
- }
-
- if (counter == 0)
- throw new InvalidOperationException ();
+ Check.Source (source);
- return initValue;
+ return Iterate (source, decimal.MinValue, (a, b) => Math.Max (a, b));
}
public static int? Max (this IEnumerable<int?> source)
{
- return IterateNullable (source, int.MinValue, (a, b) => a > b);
+ Check.Source (source);
+
+ return IterateNullable (source, (a, b) => Math.Max (a, b));
}
public static long? Max (this IEnumerable<long?> source)
{
- return IterateNullable (source, long.MinValue, (a, b) => a > b);
+ Check.Source (source);
+
+ return IterateNullable (source, (a, b) => Math.Max (a, b));
}
public static double? Max (this IEnumerable<double?> source)
{
- return IterateNullable (source, double.MinValue, (a, b) => a > b);
+ Check.Source (source);
+
+ return IterateNullable (source, (a, b) => Math.Max (a, b));
}
public static float? Max (this IEnumerable<float?> source)
{
- return IterateNullable (source, float.MinValue, (a, b) => a > b);
+ Check.Source (source);
+
+ return IterateNullable (source, (a, b) => Math.Max (a, b));
}
public static decimal? Max (this IEnumerable<decimal?> source)
{
- return IterateNullable (source, decimal.MinValue, (a, b) => a > b);
+ Check.Source (source);
+
+ return IterateNullable (source, (a, b) => Math.Max (a, b));
}
- static T? IterateNullable<T> (IEnumerable<T?> source, T initValue, Func<T?, T?, bool> selector) where T : struct
+ static T? IterateNullable<T> (IEnumerable<T?> source, Func<T, T, T> selector) where T : struct
{
- Check.SourceAndSelector (source, selector);
-
- int counter = 0;
- T? value = initValue;
+ bool empty = true;
+ T? value = null;
foreach (var element in source) {
if (!element.HasValue)
continue;
- if (selector (element.Value, value))
- value = element;
- ++counter;
+ if (!value.HasValue)
+ value = element.Value;
+ else
+ value = selector (element.Value, value.Value);
+
+ empty = false;
}
- if (counter == 0)
+ if (empty)
+ return null;
+
+ return value;
+ }
+
+ static TRet? IterateNullable<TSource, TRet> (
+ IEnumerable<TSource> source,
+ Func<TSource, TRet?> source_selector,
+ Func<TRet?, TRet?, bool> selector) where TRet : struct
+ {
+ bool empty = true;
+ TRet? value = null;
+ foreach (var element in source) {
+ TRet? item = source_selector (element);
+
+ if (!value.HasValue)
+ value = item;
+ else if (selector (item, value))
+ value = item;
+
+ empty = false;
+ }
+
+ if (empty)
return null;
return value;
public static int Max<TSource> (this IEnumerable<TSource> source, Func<TSource, int> selector)
{
- return Iterate (source, int.MinValue, (a, b) => {
- var v = selector (a); return v > b ? v : b;
- });
+ Check.SourceAndSelector (source, selector);
+
+ return Iterate (source, int.MinValue, (a, b) => Math.Max (selector (a), b));
}
public static long Max<TSource> (this IEnumerable<TSource> source, Func<TSource, long> selector)
{
- return Iterate (source, long.MinValue, (a, b) => {
- var v = selector (a); return v > b ? v : b;
- });
+ Check.SourceAndSelector (source, selector);
+
+ return Iterate (source, long.MinValue, (a, b) => Math.Max (selector (a), b));
}
public static double Max<TSource> (this IEnumerable<TSource> source, Func<TSource, double> selector)
{
- return Iterate (source, double.MinValue, (a, b) => {
- var v = selector (a); return v > b ? v : b;
- });
+ Check.SourceAndSelector (source, selector);
+
+ return Iterate (source, double.MinValue, (a, b) => Math.Max (selector (a), b));
}
public static float Max<TSource> (this IEnumerable<TSource> source, Func<TSource, float> selector)
{
- return Iterate (source, float.MinValue, (a, b) => {
- var v = selector (a); return v > b ? v : b;
- });
+ Check.SourceAndSelector (source, selector);
+
+ return Iterate (source, float.MinValue, (a, b) => Math.Max (selector (a), b));
}
public static decimal Max<TSource> (this IEnumerable<TSource> source, Func<TSource, decimal> selector)
{
- return Iterate (source, decimal.MinValue, (a, b) => {
- var v = selector (a); return v > b ? v : b;
- });
+ Check.SourceAndSelector (source, selector);
+
+ return Iterate (source, decimal.MinValue, (a, b) => Math.Max (selector (a), b));
}
static U Iterate<T, U> (IEnumerable<T> source, U initValue, Func<T, U, U> selector)
{
- Check.SourceAndSelector (source, selector);
-
- int counter = 0;
+ bool empty = true;
foreach (var element in source) {
initValue = selector (element, initValue);
- ++counter;
+ empty = false;
}
- if (counter == 0)
+ if (empty)
throw new InvalidOperationException ();
return initValue;
}
- static U? IterateNullable<T, U> (IEnumerable<T> source, U initialValue, Func<T, U?, U?> selector) where U : struct
+ public static int? Max<TSource> (this IEnumerable<TSource> source, Func<TSource, int?> selector)
{
Check.SourceAndSelector (source, selector);
- int counter = 0;
- U? value = initialValue;
- foreach (var element in source) {
- value = selector (element, value);
- if (!value.HasValue)
- continue;
-
- ++counter;
- }
-
- if (counter == 0)
- return null;
-
- return value;
- }
-
- public static int? Max<TSource> (this IEnumerable<TSource> source, Func<TSource, int?> selector)
- {
- return IterateNullable (source, int.MinValue, (a, b) => {
- var v = selector (a); return v > b ? v : b;
- });
+ return IterateNullable (source, selector, (a, b) => a > b);
}
public static long? Max<TSource> (this IEnumerable<TSource> source, Func<TSource, long?> selector)
{
- return IterateNullable (source, long.MinValue, (a, b) => {
- var v = selector (a); return v > b ? v : b;
- });
+ Check.SourceAndSelector (source, selector);
+
+ return IterateNullable (source, selector, (a, b) => a > b);
}
public static double? Max<TSource> (this IEnumerable<TSource> source, Func<TSource, double?> selector)
{
- return IterateNullable (source, double.MinValue, (a, b) => {
- var v = selector (a); return v > b ? v : b;
- });
+ Check.SourceAndSelector (source, selector);
+
+ return IterateNullable (source, selector, (a, b) => a > b);
}
public static float? Max<TSource> (this IEnumerable<TSource> source, Func<TSource, float?> selector)
{
- return IterateNullable (source, float.MinValue, (a, b) => {
- var v = selector (a); return v > b ? v : b;
- });
+ Check.SourceAndSelector (source, selector);
+
+ return IterateNullable (source, selector, (a, b) => a > b);
}
public static decimal? Max<TSource> (this IEnumerable<TSource> source, Func<TSource, decimal?> selector)
{
- return IterateNullable (source, decimal.MinValue, (a, b) => {
- var v = selector (a); return v > b ? v : b;
- });
+ Check.SourceAndSelector (source, selector);
+
+ return IterateNullable (source, selector, (a, b) => a > b);
}
- public static TResult Max<TSource, TResult> (this IEnumerable<TSource> source,
- Func<TSource, TResult> selector)
+ public static TResult Max<TSource, TResult> (this IEnumerable<TSource> source, Func<TSource, TResult> selector)
{
Check.SourceAndSelector (source, selector);
public static int Min (this IEnumerable<int> source)
{
- return Iterate (source, int.MaxValue, (a, b) => a < b);
+ Check.Source (source);
+
+ return Iterate (source, int.MaxValue, (a, b) => Math.Min (a, b));
}
public static long Min (this IEnumerable<long> source)
{
- return Iterate (source, long.MaxValue, (a, b) => a < b);
+ Check.Source (source);
+
+ return Iterate (source, long.MaxValue, (a, b) => Math.Min (a, b));
}
public static double Min (this IEnumerable<double> source)
{
- return Iterate (source, double.MaxValue, (a, b) => a < b);
+ Check.Source (source);
+
+ return Iterate (source, double.MaxValue, (a, b) => Math.Min (a, b));
}
public static float Min (this IEnumerable<float> source)
{
- return Iterate (source, float.MaxValue, (a, b) => a < b);
+ Check.Source (source);
+
+ return Iterate (source, float.MaxValue, (a, b) => Math.Min (a, b));
}
public static decimal Min (this IEnumerable<decimal> source)
{
- return Iterate (source, decimal.MaxValue, (a, b) => a < b);
+ Check.Source (source);
+
+ return Iterate (source, decimal.MaxValue, (a, b) => Math.Min (a, b));
}
public static int? Min (this IEnumerable<int?> source)
{
- return IterateNullable (source, int.MaxValue, (a, b) => a < b);
+ Check.Source (source);
+
+ return IterateNullable (source, (a, b) => Math.Min (a, b));
}
public static long? Min (this IEnumerable<long?> source)
{
- return IterateNullable (source, long.MaxValue, (a, b) => a < b);
+ Check.Source (source);
+
+ return IterateNullable (source, (a, b) => Math.Min (a, b));
}
public static double? Min (this IEnumerable<double?> source)
{
- return IterateNullable (source, double.MaxValue, (a, b) => a < b);
+ Check.Source (source);
+
+ return IterateNullable (source, (a, b) => Math.Min (a, b));
}
public static float? Min (this IEnumerable<float?> source)
{
- return IterateNullable (source, float.MaxValue, (a, b) => a < b);
+ Check.Source (source);
+
+ return IterateNullable (source, (a, b) => Math.Min (a, b));
}
public static decimal? Min (this IEnumerable<decimal?> source)
{
- return IterateNullable (source, decimal.MaxValue, (a, b) => a < b);
+ Check.Source (source);
+
+ return IterateNullable (source, (a, b) => Math.Min (a, b));
}
public static TSource Min<TSource> (this IEnumerable<TSource> source)
public static int Min<TSource> (this IEnumerable<TSource> source, Func<TSource, int> selector)
{
- return Iterate (source, int.MaxValue, (a, b) => {
- var v = selector (a); return v < b ? v : b;
- });
+ Check.SourceAndSelector (source, selector);
+
+ return Iterate (source, int.MaxValue, (a, b) => Math.Min (selector (a), b));
}
public static long Min<TSource> (this IEnumerable<TSource> source, Func<TSource, long> selector)
{
- return Iterate (source, long.MaxValue, (a, b) => {
- var v = selector (a); return v < b ? v : b;
- });
+ Check.SourceAndSelector (source, selector);
+
+ return Iterate (source, long.MaxValue, (a, b) => Math.Min (selector (a), b));
}
public static double Min<TSource> (this IEnumerable<TSource> source, Func<TSource, double> selector)
{
- return Iterate (source, double.MaxValue, (a, b) => {
- var v = selector (a); return v < b ? v : b;
- });
+ Check.SourceAndSelector (source, selector);
+
+ return Iterate (source, double.MaxValue, (a, b) => Math.Min (selector (a), b));
}
public static float Min<TSource> (this IEnumerable<TSource> source, Func<TSource, float> selector)
{
- return Iterate (source, float.MaxValue, (a, b) => {
- var v = selector (a); return v < b ? v : b;
- });
+ Check.SourceAndSelector (source, selector);
+
+ return Iterate (source, float.MaxValue, (a, b) => Math.Min (selector (a), b));
}
public static decimal Min<TSource> (this IEnumerable<TSource> source, Func<TSource, decimal> selector)
{
- return Iterate (source, decimal.MaxValue, (a, b) => {
- var v = selector (a); return v < b ? v : b;
- });
+ Check.SourceAndSelector (source, selector);
+
+ return Iterate (source, decimal.MaxValue, (a, b) => Math.Min (selector (a), b));
}
public static int? Min<TSource> (this IEnumerable<TSource> source, Func<TSource, int?> selector)
{
- return IterateNullable (source, int.MaxValue, (a, b) => {
- var v = selector (a); return v < b ? v : b;
- });
+ Check.SourceAndSelector (source, selector);
+
+ return IterateNullable (source, selector, (a, b) => a < b);
}
public static long? Min<TSource> (this IEnumerable<TSource> source, Func<TSource, long?> selector)
{
- return IterateNullable (source, long.MaxValue, (a, b) => {
- var v = selector (a); return v < b ? v : b;
- });
+ Check.SourceAndSelector (source, selector);
+
+ return IterateNullable (source, selector, (a, b) => a < b);
}
- public static double? Min<TSource> (this IEnumerable<TSource> source, Func<TSource, float?> selector)
+ public static float? Min<TSource> (this IEnumerable<TSource> source, Func<TSource, float?> selector)
{
- return IterateNullable (source, float.MaxValue, (a, b) => {
- var v = selector (a); return v < b ? v : b;
- });
+ Check.SourceAndSelector (source, selector);
+
+ return IterateNullable (source, selector, (a, b) => a < b);
}
public static double? Min<TSource> (this IEnumerable<TSource> source, Func<TSource, double?> selector)
{
- return IterateNullable (source, double.MaxValue, (a, b) => {
- var v = selector (a); return v < b ? v : b;
- });
+ Check.SourceAndSelector (source, selector);
+
+ return IterateNullable (source, selector, (a, b) => a < b);
}
public static decimal? Min<TSource> (this IEnumerable<TSource> source, Func<TSource, decimal?> selector)
{
- return IterateNullable (source, decimal.MaxValue, (a, b) => {
- var v = selector (a); return v < b ? v : b;
- });
+ Check.SourceAndSelector (source, selector);
+
+ return IterateNullable (source, selector, (a, b) => a < b);
}
- public static TResult Min<TSource, TResult> (this IEnumerable<TSource> source,
- Func<TSource, TResult> selector)
+ public static TResult Min<TSource, TResult> (this IEnumerable<TSource> source, Func<TSource, TResult> selector)
{
Check.SourceAndSelector (source, selector);
return OrderBy<TSource, TKey> (source, keySelector, null);
}
-
public static IOrderedEnumerable<TSource> OrderBy<TSource, TKey> (this IEnumerable<TSource> source,
Func<TSource, TKey> keySelector,
IComparer<TKey> comparer)
{
Check.SourceAndKeySelector (source, keySelector);
- return new InternalOrderedSequence<TSource, TKey> (
- source, keySelector, comparer, false);
+ return new OrderedSequence<TSource, TKey> (source, keySelector, comparer, SortDirection.Ascending);
}
#endregion
return OrderByDescending<TSource, TKey> (source, keySelector, null);
}
-
public static IOrderedEnumerable<TSource> OrderByDescending<TSource, TKey> (this IEnumerable<TSource> source,
Func<TSource, TKey> keySelector, IComparer<TKey> comparer)
{
Check.SourceAndKeySelector (source, keySelector);
- return new InternalOrderedSequence<TSource, TKey> (
- source, keySelector, comparer, true);
+ return new OrderedSequence<TSource, TKey> (source, keySelector, comparer, SortDirection.Descending);
}
#endregion
#endregion
-
#region Reverse
public static IEnumerable<TSource> Reverse<TSource> (this IEnumerable<TSource> source)
{
Check.Source (source);
- return CreateReverseIterator (source);
+ var list = source as IList<TSource>;
+ if (list == null)
+ list = new List<TSource> (source);
+
+ return CreateReverseIterator (list);
}
- static IEnumerable<TSource> CreateReverseIterator<TSource> (IEnumerable<TSource> source)
+ static IEnumerable<TSource> CreateReverseIterator<TSource> (IList<TSource> source)
{
- var list = new List<TSource> (source);
- list.Reverse ();
- return list;
+ for (int i = source.Count; i > 0; --i)
+ yield return source [i - 1];
}
#endregion
static IEnumerable<TResult> CreateSelectIterator<TSource, TResult> (IEnumerable<TSource> source, Func<TSource, TResult> selector)
{
- foreach (TSource element in source)
+ foreach (var element in source)
yield return selector (element);
}
#region Single
- public static TSource Single<TSource> (this IEnumerable<TSource> source)
+ static TSource Single<TSource> (this IEnumerable<TSource> source, Func<TSource, bool> predicate, Fallback fallback)
{
- Check.Source (source);
+ var found = false;
+ var item = default (TSource);
- bool otherElement = false;
- TSource singleElement = default (TSource);
- foreach (TSource element in source) {
- if (otherElement) throw new InvalidOperationException ();
- if (!otherElement) otherElement = true;
- singleElement = element;
+ foreach (var element in source) {
+ if (!predicate (element))
+ continue;
+
+ if (found)
+ throw new InvalidOperationException ();
+
+ found = true;
+ item = element;
}
- if (otherElement)
- return singleElement;
- else
+ if (!found && fallback == Fallback.Throw)
throw new InvalidOperationException ();
+
+ return item;
}
+ public static TSource Single<TSource> (this IEnumerable<TSource> source)
+ {
+ Check.Source (source);
+
+ return source.Single (PredicateOf<TSource>.Always, Fallback.Throw);
+ }
- public static TSource Single<TSource> (this IEnumerable<TSource> source,
- Func<TSource, bool> predicate)
+ public static TSource Single<TSource> (this IEnumerable<TSource> source, Func<TSource, bool> predicate)
{
Check.SourceAndPredicate (source, predicate);
- bool otherElement = false;
- TSource singleElement = default (TSource);
- foreach (TSource element in source) {
- if (predicate (element)) {
- if (otherElement) throw new InvalidOperationException ();
- if (!otherElement) otherElement = true;
- singleElement = element;
- }
- }
-
- if (otherElement)
- return singleElement;
- else
- throw new InvalidOperationException ();
+ return source.Single (predicate, Fallback.Throw);
}
#endregion
{
Check.Source (source);
- bool otherElement = false;
- TSource singleElement = default (TSource);
- foreach (TSource element in source) {
- if (otherElement) throw new InvalidOperationException ();
- if (!otherElement) otherElement = true;
- singleElement = element;
- }
-
- return singleElement;
+ return source.Single (PredicateOf<TSource>.Always, Fallback.Default);
}
-
- public static TSource SingleOrDefault<TSource> (this IEnumerable<TSource> source,
- Func<TSource, bool> predicate)
+ public static TSource SingleOrDefault<TSource> (this IEnumerable<TSource> source, Func<TSource, bool> predicate)
{
Check.SourceAndPredicate (source, predicate);
- bool otherElement = false;
- TSource singleElement = default (TSource);
- foreach (TSource element in source) {
- if (predicate (element)) {
- if (otherElement) throw new InvalidOperationException ();
- if (!otherElement) otherElement = true;
- singleElement = element;
- }
- }
-
- return singleElement;
+ return source.Single (predicate, Fallback.Default);
}
#endregion
public static int Sum (this IEnumerable<int> source)
{
- return Sum<int, int> (source, (a, b) => a + b);
+ Check.Source (source);
+
+ return Sum<int, int> (source, (a, b) => checked (a + b));
}
public static int? Sum (this IEnumerable<int?> source)
{
- return SumNullable<int?, int?> (source, (a, b) => a.HasValue ? a + b : a);
+ Check.Source (source);
+
+ return source.SumNullable<int?, int?> (0, (total, element) => element.HasValue ? checked (total + element) : total);
}
public static int Sum<TSource> (this IEnumerable<TSource> source, Func<TSource, int> selector)
{
- return Sum<TSource, int> (source, (a, b) => a + selector (b));
+ Check.SourceAndSelector (source, selector);
+
+ return Sum<TSource, int> (source, (a, b) => checked (a + selector (b)));
}
public static int? Sum<TSource> (this IEnumerable<TSource> source, Func<TSource, int?> selector)
{
- return SumNullable<TSource, int?> (source, (a, b) => {
+ Check.SourceAndSelector (source, selector);
+
+ return source.SumNullable<TSource, int?> (0, (a, b) => {
var value = selector (b);
- return value.HasValue ? a + value.Value : a;
+ return value.HasValue ? checked (a + value.Value) : a;
});
}
public static long Sum (this IEnumerable<long> source)
{
- return Sum<long, long> (source, (a, b) => a + b);
+ Check.Source (source);
+
+ return Sum<long, long> (source, (a, b) => checked (a + b));
}
public static long? Sum (this IEnumerable<long?> source)
{
- return SumNullable<long?, long?> (source, (a, b) => a.HasValue ? a + b : a);
+ Check.Source (source);
+
+ return source.SumNullable<long?, long?> (0, (total, element) => element.HasValue ? checked (total + element) : total);
}
public static long Sum<TSource> (this IEnumerable<TSource> source, Func<TSource, long> selector)
{
- return Sum<TSource, long> (source, (a, b) => a + selector (b));
+ Check.SourceAndSelector (source, selector);
+
+ return Sum<TSource, long> (source, (a, b) => checked (a + selector (b)));
}
public static long? Sum<TSource> (this IEnumerable<TSource> source, Func<TSource, long?> selector)
{
- return SumNullable<TSource, long?> (source, (a, b) => {
+ Check.SourceAndSelector (source, selector);
+
+ return source.SumNullable<TSource, long?> (0, (a, b) => {
var value = selector (b);
- return value.HasValue ? a + value.Value : a;
+ return value.HasValue ? checked (a + value.Value) : a;
});
}
public static double Sum (this IEnumerable<double> source)
{
- return Sum<double, double> (source, (a, b) => a + b);
+ Check.Source (source);
+
+ return Sum<double, double> (source, (a, b) => checked (a + b));
}
public static double? Sum (this IEnumerable<double?> source)
{
- return SumNullable<double?, double?> (source, (a, b) => a.HasValue ? a + b : a);
+ Check.Source (source);
+
+ return source.SumNullable<double?, double?> (0, (total, element) => element.HasValue ? checked (total + element) : total);
}
public static double Sum<TSource> (this IEnumerable<TSource> source, Func<TSource, double> selector)
{
- return Sum<TSource, double> (source, (a, b) => a + selector (b));
+ Check.SourceAndSelector (source, selector);
+
+ return Sum<TSource, double> (source, (a, b) => checked (a + selector (b)));
}
public static double? Sum<TSource> (this IEnumerable<TSource> source, Func<TSource, double?> selector)
{
- return SumNullable<TSource, double?> (source, (a, b) => {
+ Check.SourceAndSelector (source, selector);
+
+ return source.SumNullable<TSource, double?> (0, (a, b) => {
var value = selector (b);
- return value.HasValue ? a + value.Value : a;
+ return value.HasValue ? checked (a + value.Value) : a;
});
}
public static float Sum (this IEnumerable<float> source)
{
- return Sum<float, float> (source, (a, b) => a + b);
+ Check.Source (source);
+
+ return Sum<float, float> (source, (a, b) => checked (a + b));
}
public static float? Sum (this IEnumerable<float?> source)
{
- return SumNullable<float?, float?> (source, (a, b) => a.HasValue ? a + b : a);
+ Check.Source (source);
+
+ return source.SumNullable<float?, float?> (0, (total, element) => element.HasValue ? checked (total + element) : total);
}
public static float Sum<TSource> (this IEnumerable<TSource> source, Func<TSource, float> selector)
{
- return Sum<TSource, float> (source, (a, b) => a + selector (b));
+ Check.SourceAndSelector (source, selector);
+
+ return Sum<TSource, float> (source, (a, b) => checked (a + selector (b)));
}
public static float? Sum<TSource> (this IEnumerable<TSource> source, Func<TSource, float?> selector)
{
- return SumNullable<TSource, float?> (source, (a, b) => {
+ Check.SourceAndSelector (source, selector);
+
+ return source.SumNullable<TSource, float?> (0, (a, b) => {
var value = selector (b);
- return value.HasValue ? a + value.Value : a;
+ return value.HasValue ? checked (a + value.Value) : a;
});
}
public static decimal Sum (this IEnumerable<decimal> source)
{
- return Sum<decimal, decimal> (source, (a, b) => a + b);
+ Check.Source (source);
+
+ return Sum<decimal, decimal> (source, (a, b) => checked (a + b));
}
public static decimal? Sum (this IEnumerable<decimal?> source)
{
- return SumNullable<decimal?, decimal?> (source, (a, b) => a.HasValue ? a + b : a);
+ Check.Source (source);
+
+ return source.SumNullable<decimal?, decimal?> (0, (total, element) => element.HasValue ? checked (total + element) : total);
}
public static decimal Sum<TSource> (this IEnumerable<TSource> source, Func<TSource, decimal> selector)
{
- return Sum<TSource, decimal> (source, (a, b) => a + selector (b));
+ Check.SourceAndSelector (source, selector);
+
+ return Sum<TSource, decimal> (source, (a, b) => checked (a + selector (b)));
}
public static decimal? Sum<TSource> (this IEnumerable<TSource> source, Func<TSource, decimal?> selector)
{
- return SumNullable<TSource, decimal?> (source, (a, b) => {
+ Check.SourceAndSelector (source, selector);
+
+ return source.SumNullable<TSource, decimal?> (0, (a, b) => {
var value = selector (b);
- return value.HasValue ? a + value.Value : a;
+ return value.HasValue ? checked (a + value.Value) : a;
});
}
- static TR Sum<TA, TR> (this IEnumerable<TA> source, Func<TR, TA, TR> func)
+ static TR Sum<TA, TR> (this IEnumerable<TA> source, Func<TR, TA, TR> selector)
{
- Check.Source (source);
-
TR total = default (TR);
- int counter = 0;
+ long counter = 0;
foreach (var element in source) {
- total = func (total, element);
+ total = selector (total, element);
++counter;
}
- if (counter == 0)
- throw new InvalidOperationException ();
-
return total;
}
- static TR SumNullable<TA, TR> (this IEnumerable<TA> source, Func<TR, TA, TR> func)
+ static TR SumNullable<TA, TR> (this IEnumerable<TA> source, TR zero, Func<TR, TA, TR> selector)
{
- Check.Source (source);
-
- TR total = default (TR);
+ TR total = zero;
foreach (var element in source) {
- total = func (total, element);
+ total = selector (total, element);
}
return total;
int counter = 0;
foreach (TSource element in source) {
- if (counter++ == count)
- yield break;
-
yield return element;
+
+ if (++counter == count)
+ yield break;
}
}
static IEnumerable<TSource> CreateTakeWhileIterator<TSource> (IEnumerable<TSource> source, Func<TSource, bool> predicate)
{
- foreach (TSource element in source) {
+ foreach (var element in source) {
if (!predicate (element))
yield break;
return ThenBy<TSource, TKey> (source, keySelector, null);
}
-
public static IOrderedEnumerable<TSource> ThenBy<TSource, TKey> (this IOrderedEnumerable<TSource> source,
Func<TSource, TKey> keySelector, IComparer<TKey> comparer)
{
return ThenByDescending<TSource, TKey> (source, keySelector, null);
}
-
public static IOrderedEnumerable<TSource> ThenByDescending<TSource, TKey> (this IOrderedEnumerable<TSource> source,
Func<TSource, TKey> keySelector, IComparer<TKey> comparer)
{
#endregion
#region ToArray
+
public static TSource [] ToArray<TSource> (this IEnumerable<TSource> source)
{
Check.Source (source);
- List<TSource> list = new List<TSource> (source);
- return list.ToArray ();
+ var collection = source as ICollection<TSource>;
+ if (collection != null) {
+ var array = new TSource [collection.Count];
+ collection.CopyTo (array, 0);
+ return array;
+ }
+
+ return new List<TSource> (source).ToArray ();
}
#endregion
public static Dictionary<TKey, TSource> ToDictionary<TSource, TKey> (this IEnumerable<TSource> source,
Func<TSource, TKey> keySelector, IEqualityComparer<TKey> comparer)
{
- throw new NotImplementedException ();
- // FIXME: compiler issue
- //return ToDictionary (source, keySelector, (a) => a, comparer);
+ return ToDictionary<TSource, TKey, TSource> (source, keySelector, Function<TSource>.Identity, comparer);
}
#endregion
public static ILookup<TKey, TSource> ToLookup<TSource, TKey> (this IEnumerable<TSource> source, Func<TSource, TKey> keySelector)
{
- return ToLookup<TSource, TKey> (source, keySelector, null);
+ return ToLookup<TSource, TKey, TSource> (source, keySelector, Function<TSource>.Identity, null);
}
public static ILookup<TKey, TSource> ToLookup<TSource, TKey> (this IEnumerable<TSource> source,
Func<TSource, TKey> keySelector, IEqualityComparer<TKey> comparer)
{
- Check.SourceAndKeySelector (source, keySelector);
-
- var dictionary = new Dictionary<TKey, List<TSource>> (comparer ?? EqualityComparer<TKey>.Default);
- foreach (TSource element in source) {
- TKey key = keySelector (element);
- if (key == null)
- throw new ArgumentNullException ();
- if (!dictionary.ContainsKey (key))
- dictionary.Add (key, new List<TSource> ());
- dictionary [key].Add (element);
- }
- return new Lookup<TKey, TSource> (dictionary);
+ return ToLookup<TSource, TKey, TSource> (source, keySelector, element => element, comparer);
}
public static ILookup<TKey, TElement> ToLookup<TSource, TKey, TElement> (this IEnumerable<TSource> source,
{
Check.SourceAndKeyElementSelectors (source, keySelector, elementSelector);
- Dictionary<TKey, List<TElement>> dictionary = new Dictionary<TKey, List<TElement>> (comparer ?? EqualityComparer<TKey>.Default);
- foreach (TSource element in source) {
- TKey key = keySelector (element);
+ var dictionary = new Dictionary<TKey, List<TElement>> (comparer ?? EqualityComparer<TKey>.Default);
+ foreach (var element in source) {
+ var key = keySelector (element);
if (key == null)
- throw new ArgumentNullException ();
- if (!dictionary.ContainsKey (key))
- dictionary.Add (key, new List<TElement> ());
- dictionary [key].Add (elementSelector (element));
+ throw new ArgumentNullException ("key");
+
+ List<TElement> list;
+ if (!dictionary.TryGetValue (key, out list)) {
+ list = new List<TElement> ();
+ dictionary.Add (key, list);
+ }
+
+ list.Add (elementSelector (element));
}
+
return new Lookup<TKey, TElement> (dictionary);
}
if (comparer == null)
comparer = EqualityComparer<TSource>.Default;
- var first_enumerator = first.GetEnumerator ();
- var second_enumerator = second.GetEnumerator ();
+ using (IEnumerator<TSource> first_enumerator = first.GetEnumerator (),
+ second_enumerator = second.GetEnumerator ()) {
- while (first_enumerator.MoveNext ()) {
- if (!second_enumerator.MoveNext ())
- return false;
+ while (first_enumerator.MoveNext ()) {
+ if (!second_enumerator.MoveNext ())
+ return false;
- if (!comparer.Equals (first_enumerator.Current, second_enumerator.Current))
- return false;
- }
+ if (!comparer.Equals (first_enumerator.Current, second_enumerator.Current))
+ return false;
+ }
- return !second_enumerator.MoveNext ();
+ return !second_enumerator.MoveNext ();
+ }
}
#endregion
#endregion
+ class ReadOnlyCollectionOf<T> {
+ public static readonly ReadOnlyCollection<T> Empty = new ReadOnlyCollection<T> (new T [0]);
+ }
+
internal static ReadOnlyCollection<TSource> ToReadOnlyCollection<TSource> (this IEnumerable<TSource> source)
{
if (source == null)
- return new ReadOnlyCollection<TSource> (new TSource [0]); // could we singlotenize that?
+ return ReadOnlyCollectionOf<TSource>.Empty;
var ro = source as ReadOnlyCollection<TSource>;
if (ro != null)