-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy patharray.go
116 lines (92 loc) · 2.04 KB
/
array.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
package safe
import (
"iter"
"math"
"runtime"
"sync"
"unsafe"
)
// A generic fixed size slice with out-of-bounds protection by clamping indices to [0, len-1].
type Array[T any] struct {
lock sync.RWMutex
array []T
initialized bool
}
func NewArray[T any](length int) *Array[T] {
if length <= 0 {
// Minimum array length is 1
length = 1
} else if length > math.MaxUint16 {
// Only do a memory check for very large arrays
var m runtime.MemStats
runtime.ReadMemStats(&m)
sizeOf := uint64(unsafe.Sizeof(*new(T)))
if sizeOf == 0 {
return &Array[T]{}
}
// TODO: check if this is an accurate way calculate momory available and prevent out of memory panic
malloc := (m.Sys - m.Alloc) / sizeOf
// Limit the length by the maximum memory available
if malloc < math.MaxInt && length > int(malloc) {
length = int(malloc)
}
}
return &Array[T]{
initialized: true,
array: make([]T, length),
}
}
func (a *Array[T]) Clear() {
if !a.initialized {
return
}
a.lock.Lock()
defer a.lock.Unlock()
a.array = make([]T, a.Len())
}
func (a *Array[T]) Index(index int) T {
if !a.initialized {
return *new(T)
}
a.lock.RLock()
defer a.lock.RUnlock()
return a.array[a.clampIndex(index)]
}
func (a *Array[T]) Len() int {
if !a.initialized {
return 0
}
return len(a.array)
}
// TODO: test
func (a *Array[T]) Range() iter.Seq2[int, T] {
if !a.initialized {
return func(func(int, T) bool) {}
}
return func(yield func(int, T) bool) {
a.lock.RLock()
defer a.lock.RUnlock()
for i, v := range a.array {
if !yield(i, v) {
return
}
}
}
}
func (a *Array[T]) Set(index int, value T) {
if !a.initialized {
return
}
a.lock.Lock()
defer a.lock.Unlock()
a.array[a.clampIndex(index)] = value
}
// Clamp an index to the bounds of the array
func (a *Array[T]) clampIndex(index int) int {
if index < 0 {
return 0
} else if index >= a.Len() {
return a.Len() - 1
}
return index
}