2
2
using System ;
3
3
using System . Collections ;
4
4
using System . Linq ;
5
- using System . Diagnostics ;
6
5
7
6
namespace BrainAI . Pathfinding
8
7
{
8
+ /// <summary>
9
+ /// Zero memory allocation lookup.
10
+ /// Copied from https://github.com/ApmeM/FingerMath/blob/master/FingerMath/Collections/Lookup.cs
11
+ /// </summary>
9
12
public class Lookup < TKey , TValue > : ILookup < TKey , TValue >
10
13
{
11
14
private Dictionary < TKey , LinkedListNode < ( TKey , TValue ) > > startReference = new Dictionary < TKey , LinkedListNode < ( TKey , TValue ) > > ( ) ;
@@ -61,6 +64,7 @@ public void Add(TKey key, TValue value)
61
64
set . Add ( tuple ) ;
62
65
}
63
66
}
67
+
64
68
version ++ ;
65
69
}
66
70
@@ -82,12 +86,16 @@ public void Remove(TKey key, TValue value)
82
86
var start = startReference [ key ] ;
83
87
while (
84
88
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 ] ) )
86
91
{
87
92
start = start . next ;
88
93
}
89
94
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
+ )
91
99
{
92
100
return ;
93
101
}
@@ -139,20 +147,10 @@ public bool Contains(TKey key)
139
147
140
148
public Enumerable this [ TKey key ] => this . Find ( key ) ;
141
149
142
- public GroupingEnumerator GetEnumerator ( )
143
- {
144
- return new GroupingEnumerator ( this ) ;
145
- }
150
+ public GroupingEnumerator GetEnumerator ( ) => new GroupingEnumerator ( this ) ;
146
151
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 ) ;
153
153
154
- return new Enumerable ( startReference [ key ] , endReference [ key ] , counts [ key ] ) ;
155
- }
156
154
[ Obsolete ( "Use Find instead. This method requires boxing and allocates memory." ) ]
157
155
IEnumerable < TValue > ILookup < TKey , TValue > . this [ TKey key ] => this . Find ( key ) ;
158
156
@@ -162,19 +160,19 @@ private Enumerable Find(TKey key)
162
160
[ Obsolete ( "Use GetEnumerator instead. This method requires boxing and allocates memory." ) ]
163
161
IEnumerator < IGrouping < TKey , TValue > > IEnumerable < IGrouping < TKey , TValue > > . GetEnumerator ( ) => this . GetEnumerator ( ) ;
164
162
165
- public struct Enumerable : IEnumerable < TValue > , IEnumerable , IGrouping < TKey , TValue >
163
+ public struct Enumerable : IEnumerable < TValue > , IEnumerable , IGrouping < TKey , TValue > , ICollection < TValue >
166
164
{
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 ;
170
169
171
- public TKey Key => start . Value . Item1 ;
170
+ public bool IsReadOnly => false ;
172
171
173
- public Enumerable ( LinkedListNode < ( TKey , TValue ) > start , LinkedListNode < ( TKey , TValue ) > end , int count )
172
+ public Enumerable ( Lookup < TKey , TValue > lookup , TKey key )
174
173
{
175
- this . start = start ;
176
- this . end = end ;
177
- this . Count = count ;
174
+ this . lookup = lookup ;
175
+ this . Key = key ;
178
176
}
179
177
180
178
[ Obsolete ( "Use GetEnumerator instead. This method requires boxing and allocates memory." ) ]
@@ -183,39 +181,65 @@ public Enumerable(LinkedListNode<(TKey, TValue)> start, LinkedListNode<(TKey, TV
183
181
[ Obsolete ( "Use GetEnumerator instead. This method requires boxing and allocates memory." ) ]
184
182
IEnumerator < TValue > IEnumerable < TValue > . GetEnumerator ( ) => GetEnumerator ( ) ;
185
183
186
- public Enumerator GetEnumerator ( )
184
+ public Enumerator GetEnumerator ( ) => new Enumerator ( this . lookup , this . Key ) ;
185
+
186
+ public void Add ( TValue item )
187
187
{
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 ;
189
221
}
190
222
}
191
223
192
224
public struct Enumerator : IEnumerator < TValue > , IEnumerator
193
225
{
194
- private LinkedListNode < ( TKey , TValue ) > node ;
195
226
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 ;
198
230
private TValue current ;
199
231
200
232
public TValue Current => current ;
201
233
202
234
object IEnumerator . Current => current ;
203
235
204
- internal Enumerator ( LinkedListNode < ( TKey , TValue ) > start , LinkedListNode < ( TKey , TValue ) > end )
236
+ internal Enumerator ( Lookup < TKey , TValue > lookup , TKey key )
205
237
{
206
- this . start = start ;
207
- this . end = end ;
208
- this . node = start ;
209
238
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 ;
219
243
}
220
244
221
245
public bool MoveNext ( )
@@ -225,14 +249,14 @@ public bool MoveNext()
225
249
return false ;
226
250
}
227
251
228
- if ( version != node . List . version )
252
+ if ( version != lookup . version )
229
253
{
230
- throw new InvalidOperationException ( "EnumFailedVersion " ) ;
254
+ throw new InvalidOperationException ( "The underlying collection was changed. " ) ;
231
255
}
232
256
233
257
current = node . Value . Item2 ;
234
258
235
- if ( node == end )
259
+ if ( node == this . lookup . endReference [ key ] )
236
260
{
237
261
node = null ;
238
262
}
@@ -256,12 +280,14 @@ public void Dispose()
256
280
public struct GroupingEnumerator : IEnumerator < IGrouping < TKey , TValue > > , IEnumerator
257
281
{
258
282
private Lookup < TKey , TValue > lookup ;
283
+ private int version ;
259
284
private Dictionary < TKey , LinkedListNode < ( TKey , TValue ) > > . Enumerator bucketEnumerator ;
260
285
261
286
internal GroupingEnumerator ( Lookup < TKey , TValue > lookup )
262
287
{
263
288
this . Current = default ;
264
289
this . lookup = lookup ;
290
+ this . version = this . lookup . version ;
265
291
this . bucketEnumerator = lookup . startReference . GetEnumerator ( ) ;
266
292
}
267
293
@@ -275,11 +301,16 @@ internal GroupingEnumerator(Lookup<TKey, TValue> lookup)
275
301
276
302
public bool MoveNext ( )
277
303
{
304
+ if ( version != this . lookup . version )
305
+ {
306
+ throw new InvalidOperationException ( "The underlying collection was changed." ) ;
307
+ }
308
+
278
309
var result = this . bucketEnumerator . MoveNext ( ) ;
279
310
if ( result )
280
311
{
281
312
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 ) ;
283
314
}
284
315
return result ;
285
316
}
0 commit comments