Skip to content

Commit b4dffcf

Browse files
committed
[Update lookup to latest version from fingermath
1 parent 21aa4c7 commit b4dffcf

File tree

2 files changed

+77
-107
lines changed

2 files changed

+77
-107
lines changed

BrainAI.Tests/LookupTest.cs

-61
This file was deleted.

BrainAI/Pathfinding/Utils/Lookup.cs

+77-46
Original file line numberDiff line numberDiff line change
@@ -2,10 +2,13 @@
22
using System;
33
using System.Collections;
44
using System.Linq;
5-
using System.Diagnostics;
65

76
namespace BrainAI.Pathfinding
87
{
8+
/// <summary>
9+
/// Zero memory allocation lookup.
10+
/// Copied from https://github.com/ApmeM/FingerMath/blob/master/FingerMath/Collections/Lookup.cs
11+
/// </summary>
912
public class Lookup<TKey, TValue> : ILookup<TKey, TValue>
1013
{
1114
private Dictionary<TKey, LinkedListNode<(TKey, TValue)>> startReference = new Dictionary<TKey, LinkedListNode<(TKey, TValue)>>();
@@ -61,6 +64,7 @@ public void Add(TKey key, TValue value)
6164
set.Add(tuple);
6265
}
6366
}
67+
6468
version++;
6569
}
6670

@@ -82,12 +86,16 @@ public void Remove(TKey key, TValue value)
8286
var start = startReference[key];
8387
while (
8488
EqualityComparer<TKey>.Default.Equals(start.Value.Item1, key) &&
85-
!EqualityComparer<TValue>.Default.Equals(start.Value.Item2, value))
89+
!EqualityComparer<TValue>.Default.Equals(start.Value.Item2, value) &&
90+
!EqualityComparer<LinkedListNode<(TKey, TValue)>>.Default.Equals(start, endReference[key]))
8691
{
8792
start = start.next;
8893
}
8994

90-
if (!EqualityComparer<TKey>.Default.Equals(start.Value.Item1, key))
95+
if (
96+
!EqualityComparer<TKey>.Default.Equals(start.Value.Item1, key) ||
97+
!EqualityComparer<TValue>.Default.Equals(start.Value.Item2, value)
98+
)
9199
{
92100
return;
93101
}
@@ -139,20 +147,10 @@ public bool Contains(TKey key)
139147

140148
public Enumerable this[TKey key] => this.Find(key);
141149

142-
public GroupingEnumerator GetEnumerator()
143-
{
144-
return new GroupingEnumerator(this);
145-
}
150+
public GroupingEnumerator GetEnumerator() => new GroupingEnumerator(this);
146151

147-
private Enumerable Find(TKey key)
148-
{
149-
if (!startReference.ContainsKey(key))
150-
{
151-
return new Enumerable(null, null, 0);
152-
}
152+
private Enumerable Find(TKey key) => new Enumerable(this, key);
153153

154-
return new Enumerable(startReference[key], endReference[key], counts[key]);
155-
}
156154
[Obsolete("Use Find instead. This method requires boxing and allocates memory.")]
157155
IEnumerable<TValue> ILookup<TKey, TValue>.this[TKey key] => this.Find(key);
158156

@@ -162,19 +160,19 @@ private Enumerable Find(TKey key)
162160
[Obsolete("Use GetEnumerator instead. This method requires boxing and allocates memory.")]
163161
IEnumerator<IGrouping<TKey, TValue>> IEnumerable<IGrouping<TKey, TValue>>.GetEnumerator() => this.GetEnumerator();
164162

165-
public struct Enumerable : IEnumerable<TValue>, IEnumerable, IGrouping<TKey, TValue>
163+
public struct Enumerable : IEnumerable<TValue>, IEnumerable, IGrouping<TKey, TValue>, ICollection<TValue>
166164
{
167-
private readonly LinkedListNode<(TKey, TValue)> start;
168-
private readonly LinkedListNode<(TKey, TValue)> end;
169-
public readonly int Count;
165+
private readonly Lookup<TKey, TValue> lookup;
166+
public TKey Key { get; }
167+
168+
public int Count => this.lookup.counts.ContainsKey(this.Key) ? this.lookup.counts[this.Key] : 0;
170169

171-
public TKey Key => start.Value.Item1;
170+
public bool IsReadOnly => false;
172171

173-
public Enumerable(LinkedListNode<(TKey, TValue)> start, LinkedListNode<(TKey, TValue)> end, int count)
172+
public Enumerable(Lookup<TKey, TValue> lookup, TKey key)
174173
{
175-
this.start = start;
176-
this.end = end;
177-
this.Count = count;
174+
this.lookup = lookup;
175+
this.Key = key;
178176
}
179177

180178
[Obsolete("Use GetEnumerator instead. This method requires boxing and allocates memory.")]
@@ -183,39 +181,65 @@ public Enumerable(LinkedListNode<(TKey, TValue)> start, LinkedListNode<(TKey, TV
183181
[Obsolete("Use GetEnumerator instead. This method requires boxing and allocates memory.")]
184182
IEnumerator<TValue> IEnumerable<TValue>.GetEnumerator() => GetEnumerator();
185183

186-
public Enumerator GetEnumerator()
184+
public Enumerator GetEnumerator() => new Enumerator(this.lookup, this.Key);
185+
186+
public void Add(TValue item)
187187
{
188-
return new Enumerator(this.start, this.end);
188+
this.lookup.Add(this.Key, item);
189+
}
190+
191+
public void Clear()
192+
{
193+
this.lookup.Remove(this.Key);
194+
}
195+
196+
public bool Contains(TValue item)
197+
{
198+
foreach (var el in this)
199+
{
200+
if (EqualityComparer<TValue>.Default.Equals(el, item))
201+
{
202+
return true;
203+
}
204+
}
205+
206+
return false;
207+
}
208+
209+
public void CopyTo(TValue[] array, int arrayIndex)
210+
{
211+
foreach (var el in this)
212+
{
213+
array[arrayIndex] = el;
214+
}
215+
}
216+
217+
public bool Remove(TValue item)
218+
{
219+
this.lookup.Remove(this.Key, item);
220+
return true;
189221
}
190222
}
191223

192224
public struct Enumerator : IEnumerator<TValue>, IEnumerator
193225
{
194-
private LinkedListNode<(TKey, TValue)> node;
195226
private readonly int version;
196-
private readonly LinkedListNode<(TKey, TValue)> start;
197-
private LinkedListNode<(TKey, TValue)> end;
227+
private readonly Lookup<TKey, TValue> lookup;
228+
private readonly TKey key;
229+
private LinkedListNode<(TKey, TValue)> node;
198230
private TValue current;
199231

200232
public TValue Current => current;
201233

202234
object IEnumerator.Current => current;
203235

204-
internal Enumerator(LinkedListNode<(TKey, TValue)> start, LinkedListNode<(TKey, TValue)> end)
236+
internal Enumerator(Lookup<TKey, TValue> lookup, TKey key)
205237
{
206-
this.start = start;
207-
this.end = end;
208-
this.node = start;
209238
this.current = default;
210-
211-
if (this.node == null)
212-
{
213-
version = -1;
214-
}
215-
else
216-
{
217-
version = start.List.version;
218-
}
239+
this.version = lookup.version;
240+
this.lookup = lookup;
241+
this.key = key;
242+
this.node = this.lookup.startReference.ContainsKey(key) ? this.lookup.startReference[key] : null;
219243
}
220244

221245
public bool MoveNext()
@@ -225,14 +249,14 @@ public bool MoveNext()
225249
return false;
226250
}
227251

228-
if (version != node.List.version)
252+
if (version != lookup.version)
229253
{
230-
throw new InvalidOperationException("EnumFailedVersion");
254+
throw new InvalidOperationException("The underlying collection was changed.");
231255
}
232256

233257
current = node.Value.Item2;
234258

235-
if (node == end)
259+
if (node == this.lookup.endReference[key])
236260
{
237261
node = null;
238262
}
@@ -256,12 +280,14 @@ public void Dispose()
256280
public struct GroupingEnumerator : IEnumerator<IGrouping<TKey, TValue>>, IEnumerator
257281
{
258282
private Lookup<TKey, TValue> lookup;
283+
private int version;
259284
private Dictionary<TKey, LinkedListNode<(TKey, TValue)>>.Enumerator bucketEnumerator;
260285

261286
internal GroupingEnumerator(Lookup<TKey, TValue> lookup)
262287
{
263288
this.Current = default;
264289
this.lookup = lookup;
290+
this.version = this.lookup.version;
265291
this.bucketEnumerator = lookup.startReference.GetEnumerator();
266292
}
267293

@@ -275,11 +301,16 @@ internal GroupingEnumerator(Lookup<TKey, TValue> lookup)
275301

276302
public bool MoveNext()
277303
{
304+
if (version != this.lookup.version)
305+
{
306+
throw new InvalidOperationException("The underlying collection was changed.");
307+
}
308+
278309
var result = this.bucketEnumerator.MoveNext();
279310
if (result)
280311
{
281312
var key = this.bucketEnumerator.Current.Key;
282-
Current = new Enumerable(this.lookup.startReference[key], this.lookup.endReference[key], this.lookup.counts[key]);
313+
Current = new Enumerable(this.lookup, key);
283314
}
284315
return result;
285316
}

0 commit comments

Comments
 (0)