Distinct.cs 7.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189
  1. using System.Collections.Generic;
  2. namespace System.Linq
  3. {
  4. public static partial class EnumerableEx
  5. {
  6. /// <summary>
  7. /// Returns elements with a distinct key value by using the default equality comparer to compare key values.
  8. /// </summary>
  9. /// <typeparam name="TSource">Source sequence element type.</typeparam>
  10. /// <typeparam name="TKey">Key type.</typeparam>
  11. /// <param name="source">Source sequence.</param>
  12. /// <param name="keySelector">Key selector.</param>
  13. /// <returns>Sequence that contains the elements from the source sequence with distinct key values.</returns>
  14. public static IEnumerable<TSource> Distinct<TSource, TKey>(this IEnumerable<TSource> source, Func<TSource, TKey> keySelector)
  15. {
  16. if (source == null)
  17. {
  18. throw new ArgumentNullException(nameof(source));
  19. }
  20. if (keySelector == null)
  21. {
  22. throw new ArgumentNullException(nameof(keySelector));
  23. }
  24. return source.Distinct_(keySelector, EqualityComparer<TKey>.Default);
  25. }
  26. /// <summary>
  27. /// Returns elements with a distinct key value by using the specified equality comparer to compare key values.
  28. /// </summary>
  29. /// <typeparam name="TSource">Source sequence element type.</typeparam>
  30. /// <typeparam name="TKey">Key type.</typeparam>
  31. /// <param name="source">Source sequence.</param>
  32. /// <param name="keySelector">Key selector.</param>
  33. /// <param name="comparer">Comparer used to compare key values.</param>
  34. /// <returns>Sequence that contains the elements from the source sequence with distinct key values.</returns>
  35. public static IEnumerable<TSource> Distinct<TSource, TKey>(this IEnumerable<TSource> source, Func<TSource, TKey> keySelector, IEqualityComparer<TKey> comparer)
  36. {
  37. if (source == null)
  38. {
  39. throw new ArgumentNullException(nameof(source));
  40. }
  41. if (keySelector == null)
  42. {
  43. throw new ArgumentNullException(nameof(keySelector));
  44. }
  45. if (comparer == null)
  46. {
  47. throw new ArgumentNullException(nameof(comparer));
  48. }
  49. return source.Distinct_(keySelector, comparer);
  50. }
  51. private static IEnumerable<TSource> Distinct_<TSource, TKey>(this IEnumerable<TSource> source, Func<TSource, TKey> keySelector, IEqualityComparer<TKey> comparer)
  52. {
  53. var set = new HashSet<TKey>(comparer);
  54. foreach (var item in source)
  55. {
  56. var key = keySelector(item);
  57. if (set.Add(key))
  58. {
  59. yield return item;
  60. }
  61. }
  62. }
  63. /// <summary>
  64. /// Returns consecutive distinct elements by using the default equality comparer to compare values.
  65. /// </summary>
  66. /// <typeparam name="TSource">Source sequence element type.</typeparam>
  67. /// <param name="source">Source sequence.</param>
  68. /// <returns>Sequence without adjacent non-distinct elements.</returns>
  69. public static IEnumerable<TSource> DistinctUntilChanged<TSource>(this IEnumerable<TSource> source)
  70. {
  71. if (source == null)
  72. {
  73. throw new ArgumentNullException(nameof(source));
  74. }
  75. return source.DistinctUntilChanged_(x => x, EqualityComparer<TSource>.Default);
  76. }
  77. /// <summary>
  78. /// Returns consecutive distinct elements by using the specified equality comparer to compare values.
  79. /// </summary>
  80. /// <typeparam name="TSource">Source sequence element type.</typeparam>
  81. /// <param name="source">Source sequence.</param>
  82. /// <param name="comparer">Comparer used to compare values.</param>
  83. /// <returns>Sequence without adjacent non-distinct elements.</returns>
  84. public static IEnumerable<TSource> DistinctUntilChanged<TSource>(this IEnumerable<TSource> source, IEqualityComparer<TSource> comparer)
  85. {
  86. if (source == null)
  87. {
  88. throw new ArgumentNullException(nameof(source));
  89. }
  90. if (comparer == null)
  91. {
  92. throw new ArgumentNullException(nameof(comparer));
  93. }
  94. return source.DistinctUntilChanged_(x => x, comparer);
  95. }
  96. /// <summary>
  97. /// Returns consecutive distinct elements based on a key value by using the specified equality comparer to compare key values.
  98. /// </summary>
  99. /// <typeparam name="TSource">Source sequence element type.</typeparam>
  100. /// <typeparam name="TKey">Key type.</typeparam>
  101. /// <param name="source">Source sequence.</param>
  102. /// <param name="keySelector">Key selector.</param>
  103. /// <returns>Sequence without adjacent non-distinct elements.</returns>
  104. public static IEnumerable<TSource> DistinctUntilChanged<TSource, TKey>(this IEnumerable<TSource> source, Func<TSource, TKey> keySelector)
  105. {
  106. if (source == null)
  107. {
  108. throw new ArgumentNullException(nameof(source));
  109. }
  110. if (keySelector == null)
  111. {
  112. throw new ArgumentNullException(nameof(keySelector));
  113. }
  114. return source.DistinctUntilChanged_(keySelector, EqualityComparer<TKey>.Default);
  115. }
  116. /// <summary>
  117. /// Returns consecutive distinct elements based on a key value by using the specified equality comparer to compare key values.
  118. /// </summary>
  119. /// <typeparam name="TSource">Source sequence element type.</typeparam>
  120. /// <typeparam name="TKey">Key type.</typeparam>
  121. /// <param name="source">Source sequence.</param>
  122. /// <param name="keySelector">Key selector.</param>
  123. /// <param name="comparer">Comparer used to compare key values.</param>
  124. /// <returns>Sequence without adjacent non-distinct elements.</returns>
  125. public static IEnumerable<TSource> DistinctUntilChanged<TSource, TKey>(this IEnumerable<TSource> source, Func<TSource, TKey> keySelector, IEqualityComparer<TKey> comparer)
  126. {
  127. if (source == null)
  128. {
  129. throw new ArgumentNullException(nameof(source));
  130. }
  131. if (keySelector == null)
  132. {
  133. throw new ArgumentNullException(nameof(keySelector));
  134. }
  135. if (comparer == null)
  136. {
  137. throw new ArgumentNullException(nameof(comparer));
  138. }
  139. return source.DistinctUntilChanged_(keySelector, comparer);
  140. }
  141. private static IEnumerable<TSource> DistinctUntilChanged_<TSource, TKey>(this IEnumerable<TSource> source, Func<TSource, TKey> keySelector, IEqualityComparer<TKey> comparer)
  142. {
  143. var currentKey = default(TKey);
  144. var hasCurrentKey = false;
  145. foreach (var item in source)
  146. {
  147. var key = keySelector(item);
  148. var comparerEquals = false;
  149. if (hasCurrentKey)
  150. {
  151. comparerEquals = comparer.Equals(currentKey, key);
  152. }
  153. if (!hasCurrentKey || !comparerEquals)
  154. {
  155. hasCurrentKey = true;
  156. currentKey = key;
  157. yield return item;
  158. }
  159. }
  160. }
  161. }
  162. }