123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136 |
- // Licensed to the .NET Foundation under one or more agreements.
- // The .NET Foundation licenses this file to you under the Apache 2.0 License.
- // See the LICENSE file in the project root for more information.
- using System.Collections.Generic;
- using System.Diagnostics;
- using System.Threading;
- using System.Threading.Tasks;
- namespace System.Linq
- {
- public static partial class AsyncEnumerable
- {
- public static IAsyncEnumerable<int> Range(int start, int count)
- {
- if (count < 0)
- throw new ArgumentOutOfRangeException(nameof(count));
- var end = (long)start + count - 1L;
- if (count < 0 || end > int.MaxValue)
- throw new ArgumentOutOfRangeException(nameof(count));
- if (count == 0)
- return Empty<int>();
- return new RangeAsyncIterator(start, count);
- }
- private sealed class RangeAsyncIterator : AsyncIterator<int>, IAsyncPartition<int>
- {
- private readonly int start;
- private readonly int end;
- public RangeAsyncIterator(int start, int count)
- {
- Debug.Assert(count > 0);
- this.start = start;
- this.end = start + count;
- }
- public override AsyncIterator<int> Clone() => new RangeAsyncIterator(start, end - start);
- public Task<int> GetCountAsync(bool onlyIfCheap, CancellationToken cancellationToken) => Task.FromResult(end - start);
- public IAsyncPartition<int> Skip(int count)
- {
- var n = end - start;
- if (count >= n)
- {
- return EmptyAsyncIterator<int>.Instance;
- }
- return new RangeAsyncIterator(start + count, n - count);
- }
- public IAsyncPartition<int> Take(int count)
- {
- var n = end - start;
- if (count >= n)
- {
- return this;
- }
- return new RangeAsyncIterator(start, count);
- }
- public Task<int[]> ToArrayAsync(CancellationToken cancellationToken)
- {
- var res = new int[end - start];
- var value = start;
- for (var i = 0; i < res.Length; i++)
- {
- res[i] = value++;
- }
- return Task.FromResult(res);
- }
- public Task<List<int>> ToListAsync(CancellationToken cancellationToken)
- {
- var res = new List<int>(end - start);
- for (var value = start; value < end; value++)
- {
- res.Add(value);
- }
- return Task.FromResult(res);
- }
- public Task<Maybe<int>> TryGetElementAsync(int index, CancellationToken cancellationToken)
- {
- if ((uint)index < (uint)(end - start))
- {
- return Task.FromResult(new Maybe<int>(start + index));
- }
- return Task.FromResult(new Maybe<int>());
- }
- public Task<Maybe<int>> TryGetFirstAsync(CancellationToken cancellationToken) => Task.FromResult(new Maybe<int>(start));
- public Task<Maybe<int>> TryGetLastAsync(CancellationToken cancellationToken) => Task.FromResult(new Maybe<int>(end - 1));
- protected override async ValueTask<bool> MoveNextCore(CancellationToken cancellationToken)
- {
- switch (state)
- {
- case AsyncIteratorState.Allocated:
- current = start;
- state = AsyncIteratorState.Iterating;
- return true;
- case AsyncIteratorState.Iterating:
- current++;
- if (current != end)
- {
- return true;
- }
- break;
- }
- await DisposeAsync().ConfigureAwait(false);
- return false;
- }
- }
- }
- }
|