123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173 |
- // Copyright (c) Microsoft Open Technologies, Inc. All rights reserved. See License.txt in the project root for license information.
- using System;
- using System.Collections.Generic;
- using System.Linq;
- namespace System.Linq
- {
- /// <summary>
- /// Provides a set of additional static methods that allow querying enumerable sequences.
- /// </summary>
- public static partial class EnumerableEx
- {
- /// <summary>
- /// Determines whether an enumerable sequence is empty.
- /// </summary>
- /// <typeparam name="TSource">Source sequence element type.</typeparam>
- /// <param name="source">Source sequence.</param>
- /// <returns>true if the sequence is empty; false otherwise.</returns>
- public static bool IsEmpty<TSource>(this IEnumerable<TSource> source)
- {
- if (source == null)
- throw new ArgumentNullException("source");
- return !source.Any();
- }
- /// <summary>
- /// Returns the minimum value in the enumerable sequence by using the specified comparer to compare values.
- /// </summary>
- /// <typeparam name="TSource">Source sequence element type.</typeparam>
- /// <param name="source">Source sequence.</param>
- /// <param name="comparer">Comparer used to determine the minimum value.</param>
- /// <returns>Minimum value in the sequence.</returns>
- public static TSource Min<TSource>(this IEnumerable<TSource> source, IComparer<TSource> comparer)
- {
- if (source == null)
- throw new ArgumentNullException("source");
- if (comparer == null)
- throw new ArgumentNullException("comparer");
- return MinBy(source, x => x, comparer).First();
- }
- /// <summary>
- /// Returns the elements with the minimum key value by using the default comparer to compare key values.
- /// </summary>
- /// <typeparam name="TSource">Source sequence element type.</typeparam>
- /// <typeparam name="TKey">Key type.</typeparam>
- /// <param name="source">Source sequence.</param>
- /// <param name="keySelector">Key selector used to extract the key for each element in the sequence.</param>
- /// <returns>List with the elements that share the same minimum key value.</returns>
- public static IList<TSource> MinBy<TSource, TKey>(this IEnumerable<TSource> source, Func<TSource, TKey> keySelector)
- {
- if (source == null)
- throw new ArgumentNullException("source");
- if (keySelector == null)
- throw new ArgumentNullException("keySelector");
- return MinBy(source, keySelector, Comparer<TKey>.Default);
- }
- /// <summary>
- /// Returns the elements with the minimum key value by using the specified comparer to compare key values.
- /// </summary>
- /// <typeparam name="TSource">Source sequence element type.</typeparam>
- /// <typeparam name="TKey">Key type.</typeparam>
- /// <param name="source">Source sequence.</param>
- /// <param name="keySelector">Key selector used to extract the key for each element in the sequence.</param>
- /// <param name="comparer">Comparer used to determine the minimum key value.</param>
- /// <returns>List with the elements that share the same minimum key value.</returns>
- public static IList<TSource> MinBy<TSource, TKey>(this IEnumerable<TSource> source, Func<TSource, TKey> keySelector, IComparer<TKey> comparer)
- {
- if (source == null)
- throw new ArgumentNullException("source");
- if (keySelector == null)
- throw new ArgumentNullException("keySelector");
- if (comparer == null)
- throw new ArgumentNullException("comparer");
- return ExtremaBy(source, keySelector, (key, minValue) => -comparer.Compare(key, minValue));
- }
- /// <summary>
- /// Returns the maximum value in the enumerable sequence by using the specified comparer to compare values.
- /// </summary>
- /// <typeparam name="TSource">Source sequence element type.</typeparam>
- /// <param name="source">Source sequence.</param>
- /// <param name="comparer">Comparer used to determine the maximum value.</param>
- /// <returns>Maximum value in the sequence.</returns>
- public static TSource Max<TSource>(this IEnumerable<TSource> source, IComparer<TSource> comparer)
- {
- if (source == null)
- throw new ArgumentNullException("source");
- if (comparer == null)
- throw new ArgumentNullException("comparer");
- return MaxBy(source, x => x, comparer).First();
- }
- /// <summary>
- /// Returns the elements with the maximum key value by using the default comparer to compare key values.
- /// </summary>
- /// <typeparam name="TSource">Source sequence element type.</typeparam>
- /// <typeparam name="TKey">Key type.</typeparam>
- /// <param name="source">Source sequence.</param>
- /// <param name="keySelector">Key selector used to extract the key for each element in the sequence.</param>
- /// <returns>List with the elements that share the same maximum key value.</returns>
- public static IList<TSource> MaxBy<TSource, TKey>(this IEnumerable<TSource> source, Func<TSource, TKey> keySelector)
- {
- if (source == null)
- throw new ArgumentNullException("source");
- if (keySelector == null)
- throw new ArgumentNullException("keySelector");
- return MaxBy(source, keySelector, Comparer<TKey>.Default);
- }
- /// <summary>
- /// Returns the elements with the minimum key value by using the specified comparer to compare key values.
- /// </summary>
- /// <typeparam name="TSource">Source sequence element type.</typeparam>
- /// <typeparam name="TKey">Key type.</typeparam>
- /// <param name="source">Source sequence.</param>
- /// <param name="keySelector">Key selector used to extract the key for each element in the sequence.</param>
- /// <param name="comparer">Comparer used to determine the maximum key value.</param>
- /// <returns>List with the elements that share the same maximum key value.</returns>
- public static IList<TSource> MaxBy<TSource, TKey>(this IEnumerable<TSource> source, Func<TSource, TKey> keySelector, IComparer<TKey> comparer)
- {
- if (source == null)
- throw new ArgumentNullException("source");
- if (keySelector == null)
- throw new ArgumentNullException("keySelector");
- if (comparer == null)
- throw new ArgumentNullException("comparer");
- return ExtremaBy(source, keySelector, (key, minValue) => comparer.Compare(key, minValue));
- }
- private static IList<TSource> ExtremaBy<TSource, TKey>(IEnumerable<TSource> source, Func<TSource, TKey> keySelector, Func<TKey, TKey, int> compare)
- {
- var result = new List<TSource>();
- using (var e = source.GetEnumerator())
- {
- if (!e.MoveNext())
- throw new InvalidOperationException("Source sequence doesn't contain any elements.");
- var current = e.Current;
- var resKey = keySelector(current);
- result.Add(current);
- while (e.MoveNext())
- {
- var cur = e.Current;
- var key = keySelector(cur);
- var cmp = compare(key, resKey);
- if (cmp == 0)
- {
- result.Add(cur);
- }
- else if (cmp > 0)
- {
- result = new List<TSource> { cur };
- resKey = key;
- }
- }
- }
- return result;
- }
- }
- }
|