123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554 |
- using Cysharp.Threading.Tasks.Internal;
- using System;
- using System.Collections;
- using System.Collections.Generic;
- using System.Linq;
- using System.Threading;
- namespace Cysharp.Threading.Tasks.Linq
- {
- public static partial class UniTaskAsyncEnumerable
- {
- public static UniTask<ILookup<TKey, TSource>> ToLookupAsync<TSource, TKey>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, TKey> keySelector, CancellationToken cancellationToken = default)
- {
- Error.ThrowArgumentNullException(source, nameof(source));
- Error.ThrowArgumentNullException(keySelector, nameof(keySelector));
- return ToLookup.ToLookupAsync(source, keySelector, EqualityComparer<TKey>.Default, cancellationToken);
- }
- public static UniTask<ILookup<TKey, TSource>> ToLookupAsync<TSource, TKey>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, TKey> keySelector, IEqualityComparer<TKey> comparer, CancellationToken cancellationToken = default)
- {
- Error.ThrowArgumentNullException(source, nameof(source));
- Error.ThrowArgumentNullException(keySelector, nameof(keySelector));
- Error.ThrowArgumentNullException(comparer, nameof(comparer));
- return ToLookup.ToLookupAsync(source, keySelector, comparer, cancellationToken);
- }
- public static UniTask<ILookup<TKey, TElement>> ToLookupAsync<TSource, TKey, TElement>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, TKey> keySelector, Func<TSource, TElement> elementSelector, CancellationToken cancellationToken = default)
- {
- Error.ThrowArgumentNullException(source, nameof(source));
- Error.ThrowArgumentNullException(keySelector, nameof(keySelector));
- Error.ThrowArgumentNullException(elementSelector, nameof(elementSelector));
- return ToLookup.ToLookupAsync(source, keySelector, elementSelector, EqualityComparer<TKey>.Default, cancellationToken);
- }
- public static UniTask<ILookup<TKey, TElement>> ToLookupAsync<TSource, TKey, TElement>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, TKey> keySelector, Func<TSource, TElement> elementSelector, IEqualityComparer<TKey> comparer, CancellationToken cancellationToken = default)
- {
- Error.ThrowArgumentNullException(source, nameof(source));
- Error.ThrowArgumentNullException(keySelector, nameof(keySelector));
- Error.ThrowArgumentNullException(elementSelector, nameof(elementSelector));
- Error.ThrowArgumentNullException(comparer, nameof(comparer));
- return ToLookup.ToLookupAsync(source, keySelector, elementSelector, comparer, cancellationToken);
- }
- public static UniTask<ILookup<TKey, TSource>> ToLookupAwaitAsync<TSource, TKey>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<TKey>> keySelector, CancellationToken cancellationToken = default)
- {
- Error.ThrowArgumentNullException(source, nameof(source));
- Error.ThrowArgumentNullException(keySelector, nameof(keySelector));
- return ToLookup.ToLookupAwaitAsync(source, keySelector, EqualityComparer<TKey>.Default, cancellationToken);
- }
- public static UniTask<ILookup<TKey, TSource>> ToLookupAwaitAsync<TSource, TKey>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<TKey>> keySelector, IEqualityComparer<TKey> comparer, CancellationToken cancellationToken = default)
- {
- Error.ThrowArgumentNullException(source, nameof(source));
- Error.ThrowArgumentNullException(keySelector, nameof(keySelector));
- Error.ThrowArgumentNullException(comparer, nameof(comparer));
- return ToLookup.ToLookupAwaitAsync(source, keySelector, comparer, cancellationToken);
- }
- public static UniTask<ILookup<TKey, TElement>> ToLookupAwaitAsync<TSource, TKey, TElement>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<TKey>> keySelector, Func<TSource, UniTask<TElement>> elementSelector, CancellationToken cancellationToken = default)
- {
- Error.ThrowArgumentNullException(source, nameof(source));
- Error.ThrowArgumentNullException(keySelector, nameof(keySelector));
- Error.ThrowArgumentNullException(elementSelector, nameof(elementSelector));
- return ToLookup.ToLookupAwaitAsync(source, keySelector, elementSelector, EqualityComparer<TKey>.Default, cancellationToken);
- }
- public static UniTask<ILookup<TKey, TElement>> ToLookupAwaitAsync<TSource, TKey, TElement>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<TKey>> keySelector, Func<TSource, UniTask<TElement>> elementSelector, IEqualityComparer<TKey> comparer, CancellationToken cancellationToken = default)
- {
- Error.ThrowArgumentNullException(source, nameof(source));
- Error.ThrowArgumentNullException(keySelector, nameof(keySelector));
- Error.ThrowArgumentNullException(elementSelector, nameof(elementSelector));
- Error.ThrowArgumentNullException(comparer, nameof(comparer));
- return ToLookup.ToLookupAwaitAsync(source, keySelector, elementSelector, comparer, cancellationToken);
- }
- public static UniTask<ILookup<TKey, TSource>> ToLookupAwaitWithCancellationAsync<TSource, TKey>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<TKey>> keySelector, CancellationToken cancellationToken = default)
- {
- Error.ThrowArgumentNullException(source, nameof(source));
- Error.ThrowArgumentNullException(keySelector, nameof(keySelector));
- return ToLookup.ToLookupAwaitWithCancellationAsync(source, keySelector, EqualityComparer<TKey>.Default, cancellationToken);
- }
- public static UniTask<ILookup<TKey, TSource>> ToLookupAwaitWithCancellationAsync<TSource, TKey>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<TKey>> keySelector, IEqualityComparer<TKey> comparer, CancellationToken cancellationToken = default)
- {
- Error.ThrowArgumentNullException(source, nameof(source));
- Error.ThrowArgumentNullException(keySelector, nameof(keySelector));
- Error.ThrowArgumentNullException(comparer, nameof(comparer));
- return ToLookup.ToLookupAwaitWithCancellationAsync(source, keySelector, comparer, cancellationToken);
- }
- public static UniTask<ILookup<TKey, TElement>> ToLookupAwaitWithCancellationAsync<TSource, TKey, TElement>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<TKey>> keySelector, Func<TSource, CancellationToken, UniTask<TElement>> elementSelector, CancellationToken cancellationToken = default)
- {
- Error.ThrowArgumentNullException(source, nameof(source));
- Error.ThrowArgumentNullException(keySelector, nameof(keySelector));
- Error.ThrowArgumentNullException(elementSelector, nameof(elementSelector));
- return ToLookup.ToLookupAwaitWithCancellationAsync(source, keySelector, elementSelector, EqualityComparer<TKey>.Default, cancellationToken);
- }
- public static UniTask<ILookup<TKey, TElement>> ToLookupAwaitWithCancellationAsync<TSource, TKey, TElement>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<TKey>> keySelector, Func<TSource, CancellationToken, UniTask<TElement>> elementSelector, IEqualityComparer<TKey> comparer, CancellationToken cancellationToken = default)
- {
- Error.ThrowArgumentNullException(source, nameof(source));
- Error.ThrowArgumentNullException(keySelector, nameof(keySelector));
- Error.ThrowArgumentNullException(elementSelector, nameof(elementSelector));
- Error.ThrowArgumentNullException(comparer, nameof(comparer));
- return ToLookup.ToLookupAwaitWithCancellationAsync(source, keySelector, elementSelector, comparer, cancellationToken);
- }
- }
- internal static class ToLookup
- {
- internal static async UniTask<ILookup<TKey, TSource>> ToLookupAsync<TSource, TKey>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, TKey> keySelector, IEqualityComparer<TKey> comparer, CancellationToken cancellationToken)
- {
- var pool = ArrayPool<TSource>.Shared;
- var array = pool.Rent(16);
- var e = source.GetAsyncEnumerator(cancellationToken);
- try
- {
- var i = 0;
- while (await e.MoveNextAsync())
- {
- ArrayPoolUtil.EnsureCapacity(ref array, i, pool);
- array[i++] = e.Current;
- }
- if (i == 0)
- {
- return Lookup<TKey, TSource>.CreateEmpty();
- }
- else
- {
- return Lookup<TKey, TSource>.Create(new ArraySegment<TSource>(array, 0, i), keySelector, comparer);
- }
- }
- finally
- {
- pool.Return(array, clearArray: !RuntimeHelpersAbstraction.IsWellKnownNoReferenceContainsType<TSource>());
- if (e != null)
- {
- await e.DisposeAsync();
- }
- }
- }
- internal static async UniTask<ILookup<TKey, TElement>> ToLookupAsync<TSource, TKey, TElement>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, TKey> keySelector, Func<TSource, TElement> elementSelector, IEqualityComparer<TKey> comparer, CancellationToken cancellationToken)
- {
- var pool = ArrayPool<TSource>.Shared;
- var array = pool.Rent(16);
- IUniTaskAsyncEnumerator<TSource> e = default;
- try
- {
- e = source.GetAsyncEnumerator(cancellationToken);
- var i = 0;
- while (await e.MoveNextAsync())
- {
- ArrayPoolUtil.EnsureCapacity(ref array, i, pool);
- array[i++] = e.Current;
- }
- if (i == 0)
- {
- return Lookup<TKey, TElement>.CreateEmpty();
- }
- else
- {
- return Lookup<TKey, TElement>.Create(new ArraySegment<TSource>(array, 0, i), keySelector, elementSelector, comparer);
- }
- }
- finally
- {
- pool.Return(array, clearArray: !RuntimeHelpersAbstraction.IsWellKnownNoReferenceContainsType<TSource>());
- if (e != null)
- {
- await e.DisposeAsync();
- }
- }
- }
- // with await
- internal static async UniTask<ILookup<TKey, TSource>> ToLookupAwaitAsync<TSource, TKey>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<TKey>> keySelector, IEqualityComparer<TKey> comparer, CancellationToken cancellationToken)
- {
- var pool = ArrayPool<TSource>.Shared;
- var array = pool.Rent(16);
- IUniTaskAsyncEnumerator<TSource> e = default;
- try
- {
- e = source.GetAsyncEnumerator(cancellationToken);
- var i = 0;
- while (await e.MoveNextAsync())
- {
- ArrayPoolUtil.EnsureCapacity(ref array, i, pool);
- array[i++] = e.Current;
- }
- if (i == 0)
- {
- return Lookup<TKey, TSource>.CreateEmpty();
- }
- else
- {
- return await Lookup<TKey, TSource>.CreateAsync(new ArraySegment<TSource>(array, 0, i), keySelector, comparer);
- }
- }
- finally
- {
- pool.Return(array, clearArray: !RuntimeHelpersAbstraction.IsWellKnownNoReferenceContainsType<TSource>());
- if (e != null)
- {
- await e.DisposeAsync();
- }
- }
- }
- internal static async UniTask<ILookup<TKey, TElement>> ToLookupAwaitAsync<TSource, TKey, TElement>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<TKey>> keySelector, Func<TSource, UniTask<TElement>> elementSelector, IEqualityComparer<TKey> comparer, CancellationToken cancellationToken)
- {
- var pool = ArrayPool<TSource>.Shared;
- var array = pool.Rent(16);
- IUniTaskAsyncEnumerator<TSource> e = default;
- try
- {
- e = source.GetAsyncEnumerator(cancellationToken);
- var i = 0;
- while (await e.MoveNextAsync())
- {
- ArrayPoolUtil.EnsureCapacity(ref array, i, pool);
- array[i++] = e.Current;
- }
- if (i == 0)
- {
- return Lookup<TKey, TElement>.CreateEmpty();
- }
- else
- {
- return await Lookup<TKey, TElement>.CreateAsync(new ArraySegment<TSource>(array, 0, i), keySelector, elementSelector, comparer);
- }
- }
- finally
- {
- pool.Return(array, clearArray: !RuntimeHelpersAbstraction.IsWellKnownNoReferenceContainsType<TSource>());
- if (e != null)
- {
- await e.DisposeAsync();
- }
- }
- }
- // with cancellation
- internal static async UniTask<ILookup<TKey, TSource>> ToLookupAwaitWithCancellationAsync<TSource, TKey>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<TKey>> keySelector, IEqualityComparer<TKey> comparer, CancellationToken cancellationToken)
- {
- var pool = ArrayPool<TSource>.Shared;
- var array = pool.Rent(16);
- IUniTaskAsyncEnumerator<TSource> e = default;
- try
- {
- e = source.GetAsyncEnumerator(cancellationToken);
- var i = 0;
- while (await e.MoveNextAsync())
- {
- ArrayPoolUtil.EnsureCapacity(ref array, i, pool);
- array[i++] = e.Current;
- }
- if (i == 0)
- {
- return Lookup<TKey, TSource>.CreateEmpty();
- }
- else
- {
- return await Lookup<TKey, TSource>.CreateAsync(new ArraySegment<TSource>(array, 0, i), keySelector, comparer, cancellationToken);
- }
- }
- finally
- {
- pool.Return(array, clearArray: !RuntimeHelpersAbstraction.IsWellKnownNoReferenceContainsType<TSource>());
- if (e != null)
- {
- await e.DisposeAsync();
- }
- }
- }
- internal static async UniTask<ILookup<TKey, TElement>> ToLookupAwaitWithCancellationAsync<TSource, TKey, TElement>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<TKey>> keySelector, Func<TSource, CancellationToken, UniTask<TElement>> elementSelector, IEqualityComparer<TKey> comparer, CancellationToken cancellationToken)
- {
- var pool = ArrayPool<TSource>.Shared;
- var array = pool.Rent(16);
- IUniTaskAsyncEnumerator<TSource> e = default;
- try
- {
- e = source.GetAsyncEnumerator(cancellationToken);
- var i = 0;
- while (await e.MoveNextAsync())
- {
- ArrayPoolUtil.EnsureCapacity(ref array, i, pool);
- array[i++] = e.Current;
- }
- if (i == 0)
- {
- return Lookup<TKey, TElement>.CreateEmpty();
- }
- else
- {
- return await Lookup<TKey, TElement>.CreateAsync(new ArraySegment<TSource>(array, 0, i), keySelector, elementSelector, comparer, cancellationToken);
- }
- }
- finally
- {
- pool.Return(array, clearArray: !RuntimeHelpersAbstraction.IsWellKnownNoReferenceContainsType<TSource>());
- if (e != null)
- {
- await e.DisposeAsync();
- }
- }
- }
- // Lookup
- class Lookup<TKey, TElement> : ILookup<TKey, TElement>
- {
- static readonly Lookup<TKey, TElement> empty = new Lookup<TKey, TElement>(new Dictionary<TKey, Grouping<TKey, TElement>>());
- // original lookup keeps order but this impl does not(dictionary not guarantee)
- readonly Dictionary<TKey, Grouping<TKey, TElement>> dict;
- Lookup(Dictionary<TKey, Grouping<TKey, TElement>> dict)
- {
- this.dict = dict;
- }
- public static Lookup<TKey, TElement> CreateEmpty()
- {
- return empty;
- }
- public static Lookup<TKey, TElement> Create(ArraySegment<TElement> source, Func<TElement, TKey> keySelector, IEqualityComparer<TKey> comparer)
- {
- var dict = new Dictionary<TKey, Grouping<TKey, TElement>>(comparer);
- var arr = source.Array;
- var c = source.Count;
- for (int i = source.Offset; i < c; i++)
- {
- var key = keySelector(arr[i]);
- if (!dict.TryGetValue(key, out var list))
- {
- list = new Grouping<TKey, TElement>(key);
- dict[key] = list;
- }
- list.Add(arr[i]);
- }
- return new Lookup<TKey, TElement>(dict);
- }
- public static Lookup<TKey, TElement> Create<TSource>(ArraySegment<TSource> source, Func<TSource, TKey> keySelector, Func<TSource, TElement> elementSelector, IEqualityComparer<TKey> comparer)
- {
- var dict = new Dictionary<TKey, Grouping<TKey, TElement>>(comparer);
- var arr = source.Array;
- var c = source.Count;
- for (int i = source.Offset; i < c; i++)
- {
- var key = keySelector(arr[i]);
- var elem = elementSelector(arr[i]);
- if (!dict.TryGetValue(key, out var list))
- {
- list = new Grouping<TKey, TElement>(key);
- dict[key] = list;
- }
- list.Add(elem);
- }
- return new Lookup<TKey, TElement>(dict);
- }
- public static async UniTask<Lookup<TKey, TElement>> CreateAsync(ArraySegment<TElement> source, Func<TElement, UniTask<TKey>> keySelector, IEqualityComparer<TKey> comparer)
- {
- var dict = new Dictionary<TKey, Grouping<TKey, TElement>>(comparer);
- var arr = source.Array;
- var c = source.Count;
- for (int i = source.Offset; i < c; i++)
- {
- var key = await keySelector(arr[i]);
- if (!dict.TryGetValue(key, out var list))
- {
- list = new Grouping<TKey, TElement>(key);
- dict[key] = list;
- }
- list.Add(arr[i]);
- }
- return new Lookup<TKey, TElement>(dict);
- }
- public static async UniTask<Lookup<TKey, TElement>> CreateAsync<TSource>(ArraySegment<TSource> source, Func<TSource, UniTask<TKey>> keySelector, Func<TSource, UniTask<TElement>> elementSelector, IEqualityComparer<TKey> comparer)
- {
- var dict = new Dictionary<TKey, Grouping<TKey, TElement>>(comparer);
- var arr = source.Array;
- var c = source.Count;
- for (int i = source.Offset; i < c; i++)
- {
- var key = await keySelector(arr[i]);
- var elem = await elementSelector(arr[i]);
- if (!dict.TryGetValue(key, out var list))
- {
- list = new Grouping<TKey, TElement>(key);
- dict[key] = list;
- }
- list.Add(elem);
- }
- return new Lookup<TKey, TElement>(dict);
- }
- public static async UniTask<Lookup<TKey, TElement>> CreateAsync(ArraySegment<TElement> source, Func<TElement, CancellationToken, UniTask<TKey>> keySelector, IEqualityComparer<TKey> comparer, CancellationToken cancellationToken)
- {
- var dict = new Dictionary<TKey, Grouping<TKey, TElement>>(comparer);
- var arr = source.Array;
- var c = source.Count;
- for (int i = source.Offset; i < c; i++)
- {
- var key = await keySelector(arr[i], cancellationToken);
- if (!dict.TryGetValue(key, out var list))
- {
- list = new Grouping<TKey, TElement>(key);
- dict[key] = list;
- }
- list.Add(arr[i]);
- }
- return new Lookup<TKey, TElement>(dict);
- }
- public static async UniTask<Lookup<TKey, TElement>> CreateAsync<TSource>(ArraySegment<TSource> source, Func<TSource, CancellationToken, UniTask<TKey>> keySelector, Func<TSource, CancellationToken, UniTask<TElement>> elementSelector, IEqualityComparer<TKey> comparer, CancellationToken cancellationToken)
- {
- var dict = new Dictionary<TKey, Grouping<TKey, TElement>>(comparer);
- var arr = source.Array;
- var c = source.Count;
- for (int i = source.Offset; i < c; i++)
- {
- var key = await keySelector(arr[i], cancellationToken);
- var elem = await elementSelector(arr[i], cancellationToken);
- if (!dict.TryGetValue(key, out var list))
- {
- list = new Grouping<TKey, TElement>(key);
- dict[key] = list;
- }
- list.Add(elem);
- }
- return new Lookup<TKey, TElement>(dict);
- }
- public IEnumerable<TElement> this[TKey key] => dict.TryGetValue(key, out var g) ? g : Enumerable.Empty<TElement>();
- public int Count => dict.Count;
- public bool Contains(TKey key)
- {
- return dict.ContainsKey(key);
- }
- public IEnumerator<IGrouping<TKey, TElement>> GetEnumerator()
- {
- return dict.Values.GetEnumerator();
- }
- IEnumerator IEnumerable.GetEnumerator()
- {
- return dict.Values.GetEnumerator();
- }
- }
- class Grouping<TKey, TElement> : IGrouping<TKey, TElement> // , IUniTaskAsyncGrouping<TKey, TElement>
- {
- readonly List<TElement> elements;
- public TKey Key { get; private set; }
- public Grouping(TKey key)
- {
- this.Key = key;
- this.elements = new List<TElement>();
- }
- public void Add(TElement value)
- {
- elements.Add(value);
- }
- public IEnumerator<TElement> GetEnumerator()
- {
- return elements.GetEnumerator();
- }
- IEnumerator IEnumerable.GetEnumerator()
- {
- return elements.GetEnumerator();
- }
- public IUniTaskAsyncEnumerator<TElement> GetAsyncEnumerator(CancellationToken cancellationToken = default)
- {
- return this.ToUniTaskAsyncEnumerable().GetAsyncEnumerator(cancellationToken);
- }
- public override string ToString()
- {
- return "Key: " + Key + ", Count: " + elements.Count;
- }
- }
- }
- }
|