1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586 |
- // 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.Diagnostics.CodeAnalysis;
- // from https://github.com/dotnet/corefx/blob/ec2685715b01d12f16b08d0dfa326649b12db8ec/src/System.Linq/src/System/Linq/Set.cs
- namespace System.Linq
- {
- [ExcludeFromCodeCoverage]
- internal sealed class Set<TElement>
- {
- private readonly IEqualityComparer<TElement> _comparer;
- private int[] _buckets;
- private Slot[] _slots;
- public Set(IEqualityComparer<TElement> comparer)
- {
- _comparer = comparer ?? EqualityComparer<TElement>.Default;
- _buckets = new int[7];
- _slots = new Slot[7];
- }
- internal int Count { get; private set; }
- // If value is not in set, add it and return true; otherwise return false
- public bool Add(TElement value)
- {
- var hashCode = InternalGetHashCode(value);
- for (var i = _buckets[hashCode%_buckets.Length] - 1; i >= 0; i = _slots[i]._next)
- {
- if (_slots[i]._hashCode == hashCode && _comparer.Equals(_slots[i]._value, value))
- {
- return false;
- }
- }
- if (Count == _slots.Length)
- {
- Resize();
- }
- var index = Count;
- Count++;
- var bucket = hashCode%_buckets.Length;
- _slots[index]._hashCode = hashCode;
- _slots[index]._value = value;
- _slots[index]._next = _buckets[bucket] - 1;
- _buckets[bucket] = index + 1;
- return true;
- }
- internal int InternalGetHashCode(TElement value)
- {
- // Handle comparer implementations that throw when passed null
- return (value == null) ? 0 : _comparer.GetHashCode(value) & 0x7FFFFFFF;
- }
- private void Resize()
- {
- var newSize = checked((Count*2) + 1);
- var newBuckets = new int[newSize];
- var newSlots = new Slot[newSize];
- Array.Copy(_slots, 0, newSlots, 0, Count);
- for (var i = 0; i < Count; i++)
- {
- var bucket = newSlots[i]._hashCode%newSize;
- newSlots[i]._next = newBuckets[bucket] - 1;
- newBuckets[bucket] = i + 1;
- }
- _buckets = newBuckets;
- _slots = newSlots;
- }
- internal struct Slot
- {
- internal int _hashCode;
- internal int _next;
- internal TElement _value;
- }
- }
- }
|