| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108 | // 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;using System.Collections.Generic;using System.Linq;using System.Threading.Tasks;namespace System.Linq{    public static partial class EnumerableEx    {        /// <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(nameof(source));            if (comparer == null)                throw new ArgumentNullException(nameof(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(nameof(source));            if (keySelector == null)                throw new ArgumentNullException(nameof(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(nameof(source));            if (keySelector == null)                throw new ArgumentNullException(nameof(keySelector));            if (comparer == null)                throw new ArgumentNullException(nameof(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;        }    }}
 |