123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384 |
- // Licensed to the .NET Foundation under one or more agreements.
- // The .NET Foundation licenses this file to you under the MIT License.
- // See the LICENSE file in the project root for more information.
- using System.Collections.Generic;
- namespace System.Linq
- {
- public static partial class EnumerableEx
- {
- /// <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> MaxByWithTies<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 MaxByWithTies(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> MaxByWithTies<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;
- }
- }
- }
|