// Licensed to the .NET Foundation under one or more agreements. // The .NET Foundation licenses this file to you under the Apache 2.0 License. // See the LICENSE file in the project root for more information. using System.Collections.Generic; namespace System.Linq { public static partial class EnumerableEx { /// /// Returns the maximum value in the enumerable sequence by using the specified comparer to compare values. /// /// Source sequence element type. /// Source sequence. /// Comparer used to determine the maximum value. /// Maximum value in the sequence. public static TSource Max(this IEnumerable source, IComparer comparer) { if (source == null) { throw new ArgumentNullException(nameof(source)); } if (comparer == null) { throw new ArgumentNullException(nameof(comparer)); } return Extrema(source, x => x, comparer, 1); } /// /// Returns the elements with the maximum key value by using the default comparer to compare key values. /// /// Source sequence element type. /// Key type. /// Source sequence. /// Key selector used to extract the key for each element in the sequence. /// List with the elements that share the same maximum key value. public static IList MaxBy(this IEnumerable source, Func keySelector) { if (source == null) { throw new ArgumentNullException(nameof(source)); } if (keySelector == null) { throw new ArgumentNullException(nameof(keySelector)); } return MaxBy(source, keySelector, Comparer.Default); } /// /// Returns the elements with the minimum key value by using the specified comparer to compare key values. /// /// Source sequence element type. /// Key type. /// Source sequence. /// Key selector used to extract the key for each element in the sequence. /// Comparer used to determine the maximum key value. /// List with the elements that share the same maximum key value. public static IList MaxBy(this IEnumerable source, Func keySelector, IComparer comparer) { if (source == null) { throw new ArgumentNullException(nameof(source)); } if (keySelector == null) { throw new ArgumentNullException(nameof(keySelector)); } if (comparer == null) { throw new ArgumentNullException(nameof(comparer)); } return ExtremaBy(source, keySelector, comparer, 1); } private static TSource Extrema(IEnumerable source, Func keySelector, IComparer compare, int direction) { var result = default(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 = current; while (e.MoveNext()) { var cur = e.Current; var key = keySelector(cur); var cmp = compare.Compare(key, resKey) * direction; if (cmp == 0) { result = cur; } else if (cmp > 0) { result = cur; resKey = key; } } } return result; } private static IList ExtremaBy(IEnumerable source, Func keySelector, IComparer compare, int direction) { var result = new List(); 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.Compare(key, resKey) * direction; if (cmp == 0) { result.Add(cur); } else if (cmp > 0) { result = new List { cur }; resKey = key; } } } return result; } } }