| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119 | // Licensed to the .NET Foundation under one or more agreements.// The .NET Foundation licenses this file to you under the MIT License.// See the LICENSE file in the project root for more information. // Copied from https://github.com/dotnet/corefx/blob/f17f1e847aeab830de77f8a46656339d7b0f1b43/src/System.Linq/src/System/Linq/SingleLinkedNode.csusing System.Collections.Generic;using System.Diagnostics;namespace System.Linq{    /// <summary>    /// An immutable node in a singly-linked list of items.    /// </summary>    /// <typeparam name="TSource">The type of the node's item.</typeparam>    internal sealed class SingleLinkedNode<TSource>    {        /// <summary>        /// Constructs a tail node.        /// </summary>        /// <param name="item">The item to place in the tail node.</param>        public SingleLinkedNode(TSource item)        {            Item = item;        }        /// <summary>        /// Constructs a node linked to the specified node.        /// </summary>        /// <param name="linked">The linked node.</param>        /// <param name="item">The item to place in this node.</param>        private SingleLinkedNode(SingleLinkedNode<TSource> linked, TSource item)        {            Debug.Assert(linked != null);            Linked = linked;            Item = item;        }        /// <summary>        /// The item held by this node.        /// </summary>        public TSource Item { get; }        /// <summary>        /// The next node in the singly-linked list.        /// </summary>        public SingleLinkedNode<TSource>? Linked { get; }        /// <summary>        /// Creates a new node that holds the specified item and is linked to this node.        /// </summary>        /// <param name="item">The item to place in the new node.</param>        public SingleLinkedNode<TSource> Add(TSource item) => new SingleLinkedNode<TSource>(this, item);        /// <summary>        /// Gets the number of items in this and subsequent nodes by walking the linked list.        /// </summary>        public int GetCount()        {            var count = 0;            for (SingleLinkedNode<TSource>? node = this; node != null; node = node.Linked)            {                count++;            }            return count;        }        /// <summary>        /// Gets an <see cref="IEnumerator{TSource}"/> that enumerates the items of this node's singly-linked list in reverse.        /// </summary>        /// <param name="count">The number of items in this node.</param>        public IEnumerator<TSource> GetEnumerator(int count)        {            return ((IEnumerable<TSource>)ToArray(count)).GetEnumerator();        }        /// <summary>        /// Gets the node at a logical index by walking the linked list.        /// </summary>        /// <param name="index">The logical index.</param>        /// <remarks>        /// The caller should make sure <paramref name="index"/> is less than this node's count.        /// </remarks>        public SingleLinkedNode<TSource> GetNode(int index)        {            Debug.Assert(index >= 0 && index < GetCount());            SingleLinkedNode<TSource> node = this;            for (; index > 0; index--)            {                node = node.Linked!;                Debug.Assert(node != null);            }            return node!;        }        /// <summary>        /// Returns an <see cref="T:TSource[]"/> that contains the items of this node's singly-linked list in reverse.        /// </summary>        /// <param name="count">The number of items in this node.</param>        private TSource[] ToArray(int count)        {            Debug.Assert(count == GetCount());            var array = new TSource[count];            var index = count;            for (SingleLinkedNode<TSource>? node = this; node != null; node = node.Linked)            {                --index;                array[index] = node.Item;            }            Debug.Assert(index == 0);            return array;        }    }}
 |