// 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;
}
}
}