#region License Information /* HeuristicLab * Copyright (C) 2002-2014 Heuristic and Evolutionary Algorithms Laboratory (HEAL) * * This file is part of HeuristicLab. * * HeuristicLab is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * HeuristicLab is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with HeuristicLab. If not, see . */ #endregion using System; using System.Collections.Generic; using System.Linq; namespace HeuristicLab.Common { public static class EnumerableExtensions { /// /// Selects all elements in the sequence that are maximal with respect to the given value. /// /// /// Runtime complexity of the operation is O(N). /// /// The type of the elements. /// The enumeration in which the items with a maximal value should be found. /// The function that selects the value to compare. /// All elements in the enumeration where the selected value is the maximum. public static IEnumerable MaxItems(this IEnumerable source, Func valueSelector) { IEnumerator enumerator = source.GetEnumerator(); if (!enumerator.MoveNext()) return Enumerable.Empty(); IComparable max = valueSelector(enumerator.Current); var result = new List(); result.Add(enumerator.Current); while (enumerator.MoveNext()) { T item = enumerator.Current; IComparable comparison = valueSelector(item); if (comparison.CompareTo(max) > 0) { result.Clear(); result.Add(item); max = comparison; } else if (comparison.CompareTo(max) == 0) { result.Add(item); } } return result; } /// /// Selects all elements in the sequence that are minimal with respect to the given value. /// /// /// Runtime complexity of the operation is O(N). /// /// The type of the elements. /// The enumeration in which items with a minimal value should be found. /// The function that selects the value. /// All elements in the enumeration where the selected value is the minimum. public static IEnumerable MinItems(this IEnumerable source, Func valueSelector) { IEnumerator enumerator = source.GetEnumerator(); if (!enumerator.MoveNext()) return Enumerable.Empty(); IComparable min = valueSelector(enumerator.Current); var result = new List(); result.Add(enumerator.Current); while (enumerator.MoveNext()) { T item = enumerator.Current; IComparable comparison = valueSelector(item); if (comparison.CompareTo(min) < 0) { result.Clear(); result.Add(item); min = comparison; } else if (comparison.CompareTo(min) == 0) { result.Add(item); } } return result; } /// /// Compute the n-ary cartesian product of arbitrarily many sequences: http://blogs.msdn.com/b/ericlippert/archive/2010/06/28/computing-a-cartesian-product-with-linq.aspx /// /// The type of the elements inside each sequence /// The collection of sequences /// An enumerable sequence of all the possible combinations of elements public static IEnumerable> CartesianProduct(this IEnumerable> sequences) { IEnumerable> result = new[] { Enumerable.Empty() }; return sequences.Aggregate(result, (current, s) => (from seq in current from item in s select seq.Concat(new[] { item }))); } } }