a575963da9
Former-commit-id: da6be194a6b1221998fc28233f2503bd61dd9d14
174 lines
7.9 KiB
C#
174 lines
7.9 KiB
C#
// 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;
|
|
}
|
|
}
|
|
}
|