Grouping.cs 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490
  1. // Licensed to the .NET Foundation under one or more agreements.
  2. // The .NET Foundation licenses this file to you under the Apache 2.0 License.
  3. // See the LICENSE file in the project root for more information.
  4. using System;
  5. using System.Collections;
  6. using System.Collections.Generic;
  7. using System.Linq;
  8. using System.Runtime.ExceptionServices;
  9. using System.Threading;
  10. using System.Threading.Tasks;
  11. namespace System.Linq
  12. {
  13. public static partial class AsyncEnumerable
  14. {
  15. public static IAsyncEnumerable<IAsyncGrouping<TKey, TElement>> GroupBy<TSource, TKey, TElement>(this IAsyncEnumerable<TSource> source, Func<TSource, TKey> keySelector, Func<TSource, TElement> elementSelector, IEqualityComparer<TKey> comparer)
  16. {
  17. if (source == null)
  18. throw new ArgumentNullException(nameof(source));
  19. if (keySelector == null)
  20. throw new ArgumentNullException(nameof(keySelector));
  21. if (elementSelector == null)
  22. throw new ArgumentNullException(nameof(elementSelector));
  23. if (comparer == null)
  24. throw new ArgumentNullException(nameof(comparer));
  25. return new GroupedAsyncEnumerable<TSource, TKey, TElement>(source, keySelector, elementSelector, comparer);
  26. }
  27. public static IAsyncEnumerable<IAsyncGrouping<TKey, TElement>> GroupBy<TSource, TKey, TElement>(this IAsyncEnumerable<TSource> source, Func<TSource, TKey> keySelector, Func<TSource, TElement> elementSelector)
  28. {
  29. if (source == null)
  30. throw new ArgumentNullException(nameof(source));
  31. if (keySelector == null)
  32. throw new ArgumentNullException(nameof(keySelector));
  33. if (elementSelector == null)
  34. throw new ArgumentNullException(nameof(elementSelector));
  35. return source.GroupBy(keySelector, elementSelector, EqualityComparer<TKey>.Default);
  36. }
  37. public static IAsyncEnumerable<IAsyncGrouping<TKey, TSource>> GroupBy<TSource, TKey>(this IAsyncEnumerable<TSource> source, Func<TSource, TKey> keySelector, IEqualityComparer<TKey> comparer)
  38. {
  39. if (source == null)
  40. throw new ArgumentNullException(nameof(source));
  41. if (keySelector == null)
  42. throw new ArgumentNullException(nameof(keySelector));
  43. if (comparer == null)
  44. throw new ArgumentNullException(nameof(comparer));
  45. return new GroupedAsyncEnumerable<TSource, TKey>(source, keySelector, comparer);
  46. }
  47. public static IAsyncEnumerable<IAsyncGrouping<TKey, TSource>> GroupBy<TSource, TKey>(this IAsyncEnumerable<TSource> source, Func<TSource, TKey> keySelector)
  48. {
  49. if (source == null)
  50. throw new ArgumentNullException(nameof(source));
  51. if (keySelector == null)
  52. throw new ArgumentNullException(nameof(keySelector));
  53. return new GroupedAsyncEnumerable<TSource, TKey>(source, keySelector, EqualityComparer<TKey>.Default);
  54. }
  55. public static IAsyncEnumerable<TResult> GroupBy<TSource, TKey, TElement, TResult>(this IAsyncEnumerable<TSource> source, Func<TSource, TKey> keySelector, Func<TSource, TElement> elementSelector, Func<TKey, IAsyncEnumerable<TElement>, TResult> resultSelector, IEqualityComparer<TKey> comparer)
  56. {
  57. if (source == null)
  58. throw new ArgumentNullException(nameof(source));
  59. if (keySelector == null)
  60. throw new ArgumentNullException(nameof(keySelector));
  61. if (elementSelector == null)
  62. throw new ArgumentNullException(nameof(elementSelector));
  63. if (resultSelector == null)
  64. throw new ArgumentNullException(nameof(resultSelector));
  65. if (comparer == null)
  66. throw new ArgumentNullException(nameof(comparer));
  67. return source.GroupBy(keySelector, elementSelector, comparer)
  68. .Select(g => resultSelector(g.Key, g));
  69. }
  70. public static IAsyncEnumerable<TResult> GroupBy<TSource, TKey, TElement, TResult>(this IAsyncEnumerable<TSource> source, Func<TSource, TKey> keySelector, Func<TSource, TElement> elementSelector, Func<TKey, IAsyncEnumerable<TElement>, TResult> resultSelector)
  71. {
  72. if (source == null)
  73. throw new ArgumentNullException(nameof(source));
  74. if (keySelector == null)
  75. throw new ArgumentNullException(nameof(keySelector));
  76. if (elementSelector == null)
  77. throw new ArgumentNullException(nameof(elementSelector));
  78. if (resultSelector == null)
  79. throw new ArgumentNullException(nameof(resultSelector));
  80. return source.GroupBy(keySelector, elementSelector, EqualityComparer<TKey>.Default)
  81. .Select(g => resultSelector(g.Key, g));
  82. }
  83. public static IAsyncEnumerable<TResult> GroupBy<TSource, TKey, TResult>(this IAsyncEnumerable<TSource> source, Func<TSource, TKey> keySelector, Func<TKey, IAsyncEnumerable<TSource>, TResult> resultSelector, IEqualityComparer<TKey> comparer)
  84. {
  85. if (source == null)
  86. throw new ArgumentNullException(nameof(source));
  87. if (keySelector == null)
  88. throw new ArgumentNullException(nameof(keySelector));
  89. if (resultSelector == null)
  90. throw new ArgumentNullException(nameof(resultSelector));
  91. if (comparer == null)
  92. throw new ArgumentNullException(nameof(comparer));
  93. return new GroupedResultAsyncEnumerable<TSource, TKey, TResult>(source, keySelector, resultSelector, comparer);
  94. }
  95. public static IAsyncEnumerable<TResult> GroupBy<TSource, TKey, TResult>(this IAsyncEnumerable<TSource> source, Func<TSource, TKey> keySelector, Func<TKey, IAsyncEnumerable<TSource>, TResult> resultSelector)
  96. {
  97. if (source == null)
  98. throw new ArgumentNullException(nameof(source));
  99. if (keySelector == null)
  100. throw new ArgumentNullException(nameof(keySelector));
  101. if (resultSelector == null)
  102. throw new ArgumentNullException(nameof(resultSelector));
  103. return GroupBy(source, keySelector, resultSelector, EqualityComparer<TKey>.Default);
  104. }
  105. private static IEnumerable<IGrouping<TKey, TElement>> GroupUntil<TSource, TKey, TElement>(this IEnumerable<TSource> source, Func<TSource, TKey> keySelector, Func<TSource, TElement> elementSelector, IComparer<TKey> comparer)
  106. {
  107. var group = default(EnumerableGrouping<TKey, TElement>);
  108. foreach (var x in source)
  109. {
  110. var key = keySelector(x);
  111. if (group == null || comparer.Compare(group.Key, key) != 0)
  112. {
  113. group = new EnumerableGrouping<TKey, TElement>(key);
  114. yield return group;
  115. }
  116. group.Add(elementSelector(x));
  117. }
  118. }
  119. internal sealed class GroupedResultAsyncEnumerable<TSource, TKey, TResult> : IIListProvider<TResult>
  120. {
  121. private readonly IAsyncEnumerable<TSource> source;
  122. private readonly Func<TSource, TKey> keySelector;
  123. private readonly Func<TKey, IAsyncEnumerable<TSource>, TResult> resultSelector;
  124. private readonly IEqualityComparer<TKey> comparer;
  125. public GroupedResultAsyncEnumerable(IAsyncEnumerable<TSource> source, Func<TSource, TKey> keySelector, Func<TKey, IAsyncEnumerable<TSource>, TResult> resultSelector, IEqualityComparer<TKey> comparer)
  126. {
  127. if (source == null) throw new ArgumentNullException(nameof(source));
  128. if (keySelector == null) throw new ArgumentNullException(nameof(keySelector));
  129. if (resultSelector == null) throw new ArgumentNullException(nameof(resultSelector));
  130. this.source = source;
  131. this.keySelector = keySelector;
  132. this.resultSelector = resultSelector;
  133. this.comparer = comparer;
  134. }
  135. public IAsyncEnumerator<TResult> GetEnumerator()
  136. {
  137. Internal.Lookup<TKey, TSource> lookup = null;
  138. IEnumerator<TResult> enumerator = null;
  139. return CreateEnumerator(
  140. async ct =>
  141. {
  142. if (lookup == null)
  143. {
  144. lookup = await Internal.Lookup<TKey, TSource>.CreateAsync(source, keySelector, comparer, ct).ConfigureAwait(false);
  145. enumerator = lookup.ApplyResultSelector(resultSelector).GetEnumerator();
  146. }
  147. // By the time we get here, the lookup is sync
  148. if (ct.IsCancellationRequested)
  149. return false;
  150. return enumerator?.MoveNext() ?? false;
  151. },
  152. () => enumerator.Current,
  153. () =>
  154. {
  155. if (enumerator != null)
  156. {
  157. enumerator.Dispose();
  158. enumerator = null;
  159. }
  160. });
  161. }
  162. public async Task<TResult[]> ToArrayAsync(CancellationToken cancellationToken)
  163. {
  164. var lookup = await Internal.Lookup<TKey, TSource>.CreateAsync(source, keySelector, comparer, cancellationToken).ConfigureAwait(false);
  165. return lookup.ToArray(resultSelector);
  166. }
  167. public async Task<List<TResult>> ToListAsync(CancellationToken cancellationToken)
  168. {
  169. var lookup = await Internal.Lookup<TKey, TSource>.CreateAsync(source, keySelector, comparer, cancellationToken).ConfigureAwait(false);
  170. return lookup.ToList(resultSelector);
  171. }
  172. public async Task<int> GetCountAsync(bool onlyIfCheap, CancellationToken cancellationToken)
  173. {
  174. if (onlyIfCheap)
  175. {
  176. return -1;
  177. }
  178. var lookup = await Internal.Lookup<TKey, TSource>.CreateAsync(source, keySelector, comparer, cancellationToken).ConfigureAwait(false);
  179. return lookup.Count;
  180. }
  181. }
  182. internal sealed class GroupedAsyncEnumerable<TSource, TKey, TElement> : IIListProvider<IAsyncGrouping<TKey, TElement>>
  183. {
  184. private readonly IAsyncEnumerable<TSource> source;
  185. private readonly Func<TSource, TKey> keySelector;
  186. private readonly Func<TSource, TElement> elementSelector;
  187. private readonly IEqualityComparer<TKey> comparer;
  188. public GroupedAsyncEnumerable(IAsyncEnumerable<TSource> source, Func<TSource, TKey> keySelector, Func<TSource, TElement> elementSelector, IEqualityComparer<TKey> comparer)
  189. {
  190. if (source == null) throw new ArgumentNullException(nameof(source));
  191. if (keySelector == null) throw new ArgumentNullException(nameof(keySelector));
  192. if (elementSelector == null) throw new ArgumentNullException(nameof(elementSelector));
  193. this.source = source;
  194. this.keySelector = keySelector;
  195. this.elementSelector = elementSelector;
  196. this.comparer = comparer;
  197. }
  198. public IAsyncEnumerator<IAsyncGrouping<TKey, TElement>> GetEnumerator()
  199. {
  200. Internal.Lookup<TKey, TElement> lookup = null;
  201. IEnumerator<IGrouping<TKey, TElement>> enumerator = null;
  202. return CreateEnumerator(
  203. async ct =>
  204. {
  205. if (lookup == null)
  206. {
  207. lookup = await Internal.Lookup<TKey, TElement>.CreateAsync(source, keySelector, elementSelector, comparer, ct).ConfigureAwait(false);
  208. enumerator = lookup.GetEnumerator();
  209. }
  210. // By the time we get here, the lookup is sync
  211. if (ct.IsCancellationRequested)
  212. return false;
  213. return enumerator?.MoveNext() ?? false;
  214. },
  215. () => (IAsyncGrouping<TKey, TElement>)enumerator?.Current,
  216. () =>
  217. {
  218. if (enumerator != null)
  219. {
  220. enumerator.Dispose();
  221. enumerator = null;
  222. }
  223. });
  224. }
  225. public async Task<IAsyncGrouping<TKey, TElement>[]> ToArrayAsync(CancellationToken cancellationToken)
  226. {
  227. IIListProvider<IAsyncGrouping<TKey, TElement>> lookup = await Internal.Lookup<TKey, TElement>.CreateAsync(source, keySelector, elementSelector, comparer, cancellationToken).ConfigureAwait(false);
  228. return await lookup.ToArrayAsync(cancellationToken).ConfigureAwait(false);
  229. }
  230. public async Task<List<IAsyncGrouping<TKey, TElement>>> ToListAsync(CancellationToken cancellationToken)
  231. {
  232. IIListProvider<IAsyncGrouping<TKey, TElement>> lookup = await Internal.Lookup<TKey, TElement>.CreateAsync(source, keySelector, elementSelector, comparer, cancellationToken).ConfigureAwait(false);
  233. return await lookup.ToListAsync(cancellationToken).ConfigureAwait(false);
  234. }
  235. public async Task<int> GetCountAsync(bool onlyIfCheap, CancellationToken cancellationToken)
  236. {
  237. if (onlyIfCheap)
  238. {
  239. return -1;
  240. }
  241. var lookup = await Internal.Lookup<TKey, TElement>.CreateAsync(source, keySelector, elementSelector, comparer, cancellationToken).ConfigureAwait(false);
  242. return lookup.Count;
  243. }
  244. }
  245. internal sealed class GroupedAsyncEnumerable<TSource, TKey> : IIListProvider<IAsyncGrouping<TKey, TSource>>
  246. {
  247. private readonly IAsyncEnumerable<TSource> source;
  248. private readonly Func<TSource, TKey> keySelector;
  249. private readonly IEqualityComparer<TKey> comparer;
  250. public GroupedAsyncEnumerable(IAsyncEnumerable<TSource> source, Func<TSource, TKey> keySelector, IEqualityComparer<TKey> comparer)
  251. {
  252. if (source == null) throw new ArgumentNullException(nameof(source));
  253. if (keySelector == null) throw new ArgumentNullException(nameof(keySelector));
  254. this.source = source;
  255. this.keySelector = keySelector;
  256. this.comparer = comparer;
  257. }
  258. public IAsyncEnumerator<IAsyncGrouping<TKey, TSource>> GetEnumerator()
  259. {
  260. Internal.Lookup<TKey, TSource> lookup = null;
  261. IEnumerator<IGrouping<TKey, TSource>> enumerator = null;
  262. return CreateEnumerator(
  263. async ct =>
  264. {
  265. if (lookup == null)
  266. {
  267. lookup = await Internal.Lookup<TKey, TSource>.CreateAsync(source, keySelector, comparer, ct).ConfigureAwait(false);
  268. enumerator = lookup.GetEnumerator();
  269. }
  270. // By the time we get here, the lookup is sync
  271. if (ct.IsCancellationRequested)
  272. return false;
  273. return enumerator?.MoveNext() ?? false;
  274. },
  275. () => (IAsyncGrouping<TKey, TSource>)enumerator?.Current,
  276. () =>
  277. {
  278. if (enumerator != null)
  279. {
  280. enumerator.Dispose();
  281. enumerator = null;
  282. }
  283. });
  284. }
  285. public async Task<IAsyncGrouping<TKey, TSource>[]> ToArrayAsync(CancellationToken cancellationToken)
  286. {
  287. IIListProvider<IAsyncGrouping<TKey, TSource>> lookup = await Internal.Lookup<TKey, TSource>.CreateAsync(source, keySelector, comparer, cancellationToken).ConfigureAwait(false);
  288. return await lookup.ToArrayAsync(cancellationToken).ConfigureAwait(false);
  289. }
  290. public async Task<List<IAsyncGrouping<TKey, TSource>>> ToListAsync(CancellationToken cancellationToken)
  291. {
  292. IIListProvider<IAsyncGrouping<TKey, TSource>> lookup = await Internal.Lookup<TKey, TSource>.CreateAsync(source, keySelector, comparer, cancellationToken).ConfigureAwait(false);
  293. return await lookup.ToListAsync(cancellationToken).ConfigureAwait(false);
  294. }
  295. public async Task<int> GetCountAsync(bool onlyIfCheap, CancellationToken cancellationToken)
  296. {
  297. if (onlyIfCheap)
  298. {
  299. return -1;
  300. }
  301. var lookup = await Internal.Lookup<TKey, TSource>.CreateAsync(source, keySelector, comparer, cancellationToken).ConfigureAwait(false);
  302. return lookup.Count;
  303. }
  304. }
  305. }
  306. }
  307. // Note: The type here has to be internal as System.Linq has it's own public copy we're not using
  308. namespace System.Linq.Internal
  309. {
  310. /// Adapted from System.Linq.Grouping from .NET Framework
  311. /// Source: https://github.com/dotnet/corefx/blob/b90532bc97b07234a7d18073819d019645285f1c/src/System.Linq/src/System/Linq/Grouping.cs#L64
  312. internal class Grouping<TKey, TElement> : IGrouping<TKey, TElement>, IList<TElement>, IAsyncGrouping<TKey, TElement>
  313. {
  314. internal int _count;
  315. internal TElement[] _elements;
  316. internal int _hashCode;
  317. internal Grouping<TKey, TElement> _hashNext;
  318. internal TKey _key;
  319. internal Grouping<TKey, TElement> _next;
  320. IEnumerator IEnumerable.GetEnumerator()
  321. {
  322. return GetEnumerator();
  323. }
  324. public IEnumerator<TElement> GetEnumerator()
  325. {
  326. for (var i = 0; i < _count; i++)
  327. {
  328. yield return _elements[i];
  329. }
  330. }
  331. // DDB195907: implement IGrouping<>.Key implicitly
  332. // so that WPF binding works on this property.
  333. public TKey Key
  334. {
  335. get { return _key; }
  336. }
  337. int ICollection<TElement>.Count
  338. {
  339. get { return _count; }
  340. }
  341. bool ICollection<TElement>.IsReadOnly
  342. {
  343. get { return true; }
  344. }
  345. void ICollection<TElement>.Add(TElement item)
  346. {
  347. throw new NotSupportedException(Strings.NOT_SUPPORTED);
  348. }
  349. void ICollection<TElement>.Clear()
  350. {
  351. throw new NotSupportedException(Strings.NOT_SUPPORTED);
  352. }
  353. bool ICollection<TElement>.Contains(TElement item)
  354. {
  355. return Array.IndexOf(_elements, item, 0, _count) >= 0;
  356. }
  357. void ICollection<TElement>.CopyTo(TElement[] array, int arrayIndex)
  358. {
  359. Array.Copy(_elements, 0, array, arrayIndex, _count);
  360. }
  361. bool ICollection<TElement>.Remove(TElement item)
  362. {
  363. throw new NotSupportedException(Strings.NOT_SUPPORTED);
  364. }
  365. int IList<TElement>.IndexOf(TElement item)
  366. {
  367. return Array.IndexOf(_elements, item, 0, _count);
  368. }
  369. void IList<TElement>.Insert(int index, TElement item)
  370. {
  371. throw new NotSupportedException(Strings.NOT_SUPPORTED);
  372. }
  373. void IList<TElement>.RemoveAt(int index)
  374. {
  375. throw new NotSupportedException(Strings.NOT_SUPPORTED);
  376. }
  377. TElement IList<TElement>.this[int index]
  378. {
  379. get
  380. {
  381. if (index < 0 || index >= _count)
  382. {
  383. throw new ArgumentOutOfRangeException(nameof(index));
  384. }
  385. return _elements[index];
  386. }
  387. set { throw new NotSupportedException(Strings.NOT_SUPPORTED); }
  388. }
  389. internal void Add(TElement element)
  390. {
  391. if (_elements.Length == _count)
  392. {
  393. Array.Resize(ref _elements, checked(_count*2));
  394. }
  395. _elements[_count] = element;
  396. _count++;
  397. }
  398. internal void Trim()
  399. {
  400. if (_elements.Length != _count)
  401. {
  402. Array.Resize(ref _elements, _count);
  403. }
  404. }
  405. IAsyncEnumerator<TElement> IAsyncEnumerable<TElement>.GetEnumerator()
  406. {
  407. return this.ToAsyncEnumerable().GetEnumerator();
  408. }
  409. }
  410. }