loudo-ds-lmap 0.0.4 → 0.0.8
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/index.d.ts +21 -14
- package/dist/index.js +89 -64
- package/package.json +3 -2
package/dist/index.d.ts
CHANGED
@@ -8,31 +8,38 @@ interface Bucket<K, V> {
|
|
8
8
|
chain: B<K, V>;
|
9
9
|
}
|
10
10
|
type B<K, V> = Bucket<K, V> | null;
|
11
|
-
export interface
|
11
|
+
export interface Config<K extends {}, V extends {}> {
|
12
12
|
readonly load?: number;
|
13
|
-
|
14
|
-
|
15
|
-
|
13
|
+
readonly initial?: number;
|
14
|
+
readonly hashCode: (key: K) => number;
|
15
|
+
readonly keyEq?: (key1: K, key2: K) => boolean;
|
16
|
+
readonly valueEq?: (value1: V, value2: V) => boolean;
|
16
17
|
}
|
18
|
+
type Conf<K extends {}, V extends {}> = Required<Config<K, V>>;
|
19
|
+
export declare const conf: unique symbol;
|
20
|
+
export declare const firstB: unique symbol;
|
21
|
+
export declare const lastB: unique symbol;
|
22
|
+
export declare const buckets: unique symbol;
|
23
|
+
export declare const size: unique symbol;
|
17
24
|
export declare class LMap<K extends {}, V extends {}> {
|
18
|
-
protected
|
19
|
-
protected buckets: B<K, V>[];
|
20
|
-
protected firstB: B<K, V>;
|
21
|
-
protected lastB: B<K, V>;
|
22
|
-
protected
|
23
|
-
constructor(input: MapInput<K, V>,
|
25
|
+
protected [conf]: Conf<K, V>;
|
26
|
+
protected [buckets]: B<K, V>[];
|
27
|
+
protected [firstB]: B<K, V>;
|
28
|
+
protected [lastB]: B<K, V>;
|
29
|
+
protected [size]: number;
|
30
|
+
constructor(input: MapInput<K, V>, _config: Config<K, V>);
|
31
|
+
static from<K extends {}, V extends {}>(input: MapInput<K, V>, config: Config<K, V>): LMap<K, V>;
|
24
32
|
get size(): number;
|
25
|
-
get
|
33
|
+
get capacity(): number;
|
26
34
|
get first(): Entry<K, V> | undefined;
|
27
35
|
get last(): Entry<K, V> | undefined;
|
28
36
|
get only(): Entry<K, V>;
|
29
37
|
get keyEq(): (k1: K, k2: K) => boolean;
|
30
38
|
get valueEq(): (v1: V, v2: V) => boolean;
|
31
|
-
get load(): number;
|
32
39
|
hasKey(key: K): boolean;
|
33
40
|
get(key: K): V | undefined;
|
34
|
-
protected afterGet(
|
35
|
-
[Symbol.iterator]():
|
41
|
+
protected afterGet(_bucket: B<K, V>): void;
|
42
|
+
[Symbol.iterator](): Iterator<Entry<K, V>>;
|
36
43
|
private grow;
|
37
44
|
private bucket;
|
38
45
|
put(key: K, value: V): V | undefined;
|
package/dist/index.js
CHANGED
@@ -1,29 +1,53 @@
|
|
1
|
-
|
1
|
+
var _a, _b, _c, _d;
|
2
|
+
import { OnlyError } from "loudo-ds-core";
|
2
3
|
import { MapChange, MapRemove } from "loudo-ds-map-interfaces";
|
4
|
+
import { One } from "loudo-ds-one";
|
5
|
+
import { mixin } from "loudo-mixin";
|
6
|
+
function confize(config) {
|
7
|
+
var _e, _f, _g, _h;
|
8
|
+
return {
|
9
|
+
load: (_e = config.load) !== null && _e !== void 0 ? _e : 0.75,
|
10
|
+
initial: (_f = config.initial) !== null && _f !== void 0 ? _f : 4,
|
11
|
+
keyEq: (_g = config.keyEq) !== null && _g !== void 0 ? _g : Object.is,
|
12
|
+
valueEq: (_h = config.valueEq) !== null && _h !== void 0 ? _h : Object.is,
|
13
|
+
hashCode: config.hashCode
|
14
|
+
};
|
15
|
+
}
|
16
|
+
export const conf = Symbol("conf");
|
17
|
+
export const firstB = Symbol("firstB");
|
18
|
+
export const lastB = Symbol("lastB");
|
19
|
+
export const buckets = Symbol("buckets");
|
20
|
+
export const size = Symbol("size");
|
3
21
|
export class LMap {
|
4
|
-
constructor(input,
|
5
|
-
this
|
6
|
-
this
|
7
|
-
this
|
8
|
-
this
|
9
|
-
|
10
|
-
|
11
|
-
|
12
|
-
|
22
|
+
constructor(input, _config) {
|
23
|
+
this[_a] = [];
|
24
|
+
this[_b] = null;
|
25
|
+
this[_c] = null;
|
26
|
+
this[_d] = 0;
|
27
|
+
const c = confize(_config);
|
28
|
+
if (c.load <= 0 || c.load >= 1)
|
29
|
+
throw new TypeError("illegal load factor " + c.load);
|
30
|
+
if (!Number.isSafeInteger(c.initial) || c.initial <= 0)
|
31
|
+
throw new TypeError("illegal capacity " + c.initial);
|
32
|
+
this[conf] = c;
|
33
|
+
this[buckets] = new Array(c.initial);
|
34
|
+
this[buckets].fill(null);
|
13
35
|
this.putAll(input);
|
14
36
|
}
|
15
|
-
|
16
|
-
|
17
|
-
|
18
|
-
get
|
37
|
+
static from(input, config) {
|
38
|
+
return new LMap(input, config);
|
39
|
+
}
|
40
|
+
get size() { return this[size]; }
|
41
|
+
get capacity() { return this[buckets].length; }
|
42
|
+
get first() { var _e; return (_e = this[firstB]) !== null && _e !== void 0 ? _e : undefined; }
|
43
|
+
get last() { var _e; return (_e = this[lastB]) !== null && _e !== void 0 ? _e : undefined; }
|
19
44
|
get only() {
|
20
|
-
if (this
|
45
|
+
if (this[size] !== 1)
|
21
46
|
throw new OnlyError();
|
22
|
-
return this
|
47
|
+
return this[firstB];
|
23
48
|
}
|
24
|
-
get keyEq() {
|
25
|
-
get valueEq() {
|
26
|
-
get load() { var _a; return (_a = this.config.load) !== null && _a !== void 0 ? _a : 0.75; }
|
49
|
+
get keyEq() { return this[conf].keyEq; }
|
50
|
+
get valueEq() { return this[conf].valueEq; }
|
27
51
|
hasKey(key) {
|
28
52
|
return this.bucket(key) !== null;
|
29
53
|
}
|
@@ -32,27 +56,27 @@ export class LMap {
|
|
32
56
|
this.afterGet(bucket);
|
33
57
|
return bucket === null || bucket === void 0 ? void 0 : bucket.value;
|
34
58
|
}
|
35
|
-
afterGet(
|
36
|
-
*[Symbol.iterator]() {
|
37
|
-
for (let bucket = this
|
59
|
+
afterGet(_bucket) { }
|
60
|
+
*[(_a = buckets, _b = firstB, _c = lastB, _d = size, Symbol.iterator)]() {
|
61
|
+
for (let bucket = this[firstB]; bucket !== null; bucket = bucket.next) {
|
38
62
|
yield bucket;
|
39
63
|
}
|
40
64
|
}
|
41
65
|
grow() {
|
42
|
-
const
|
43
|
-
|
44
|
-
|
45
|
-
this
|
46
|
-
for (let bucket = this
|
47
|
-
const i = Math.abs(bucket.hashCode) %
|
48
|
-
const chain =
|
49
|
-
|
50
|
-
bucket.chain = chain;
|
66
|
+
const old = this[buckets];
|
67
|
+
const n = new Array(old.length * 2);
|
68
|
+
n.fill(null);
|
69
|
+
this[buckets] = n;
|
70
|
+
for (let bucket = this[firstB]; bucket != null; bucket = bucket.next) {
|
71
|
+
const i = Math.abs(bucket.hashCode) % n.length;
|
72
|
+
const chain = n[i];
|
73
|
+
n[i] = bucket;
|
74
|
+
bucket.chain = chain !== null && chain !== void 0 ? chain : null;
|
51
75
|
}
|
52
76
|
}
|
53
77
|
bucket(key) {
|
54
|
-
const hash = this.
|
55
|
-
let bucket = this
|
78
|
+
const hash = this[conf].hashCode(key);
|
79
|
+
let bucket = this[buckets][Math.abs(hash) % this[buckets].length];
|
56
80
|
while (bucket !== null) {
|
57
81
|
if (bucket.hashCode === hash && this.keyEq(key, bucket.key))
|
58
82
|
return bucket;
|
@@ -61,49 +85,48 @@ export class LMap {
|
|
61
85
|
return null;
|
62
86
|
}
|
63
87
|
put(key, value) {
|
64
|
-
if (this
|
88
|
+
if (this[size] / this[buckets].length >= this[conf].load)
|
65
89
|
this.grow();
|
66
|
-
const hashCode = this.
|
90
|
+
const hashCode = this[conf].hashCode(key);
|
67
91
|
const bucket = {
|
68
92
|
key, value, hashCode, prev: null, next: null, chain: null,
|
69
93
|
};
|
70
|
-
const i = Math.abs(hashCode) % this
|
71
|
-
let b = this
|
94
|
+
const i = Math.abs(hashCode) % this[buckets].length;
|
95
|
+
let b = this[buckets][i];
|
72
96
|
while (b !== null) {
|
73
97
|
if (b.hashCode === hashCode && this.keyEq(key, b.key)) {
|
74
98
|
const r = b.value;
|
75
99
|
b.value = value;
|
76
100
|
this.afterGet(b);
|
77
|
-
const elements = toTin([bucket]);
|
78
101
|
this.fire({
|
79
102
|
cleared: false,
|
80
|
-
added: { elements:
|
81
|
-
removed: { elements:
|
103
|
+
added: { elements: One.of(bucket), at: undefined },
|
104
|
+
removed: { elements: One.of({ key: key, value: r }), at: undefined }
|
82
105
|
});
|
83
106
|
return r;
|
84
107
|
}
|
85
108
|
b = b.chain;
|
86
109
|
}
|
87
|
-
bucket.chain = this
|
88
|
-
this
|
89
|
-
if (this
|
90
|
-
this
|
91
|
-
this
|
110
|
+
bucket.chain = this[buckets][i];
|
111
|
+
this[buckets][i] = bucket;
|
112
|
+
if (this[firstB] === null) {
|
113
|
+
this[firstB] = bucket;
|
114
|
+
this[lastB] = bucket;
|
92
115
|
}
|
93
116
|
else {
|
94
|
-
this
|
95
|
-
bucket.prev = this
|
96
|
-
this
|
117
|
+
this[lastB].next = bucket;
|
118
|
+
bucket.prev = this[lastB];
|
119
|
+
this[lastB] = bucket;
|
97
120
|
}
|
98
|
-
this
|
99
|
-
this.fire({ cleared: false, added: { elements:
|
121
|
+
this[size]++;
|
122
|
+
this.fire({ cleared: false, added: { elements: One.of(bucket), at: undefined } });
|
100
123
|
return undefined;
|
101
124
|
}
|
102
125
|
removeKey(key) {
|
103
|
-
const hash = this.
|
126
|
+
const hash = this[conf].hashCode(key);
|
104
127
|
let prev = null;
|
105
|
-
const i = Math.abs(hash) % this
|
106
|
-
let bucket = this
|
128
|
+
const i = Math.abs(hash) % this[buckets].length;
|
129
|
+
let bucket = this[buckets][i];
|
107
130
|
while (bucket !== null) {
|
108
131
|
if (bucket.hashCode === hash && this.keyEq(key, bucket.key))
|
109
132
|
break;
|
@@ -112,31 +135,33 @@ export class LMap {
|
|
112
135
|
}
|
113
136
|
if (bucket === null)
|
114
137
|
return undefined;
|
115
|
-
this
|
138
|
+
this[size]--;
|
116
139
|
if (prev === null)
|
117
|
-
this
|
140
|
+
this[buckets][i] = bucket.chain;
|
118
141
|
else
|
119
142
|
prev.chain = bucket.chain;
|
120
|
-
if (this
|
121
|
-
this
|
122
|
-
if (this
|
123
|
-
this
|
143
|
+
if (this[firstB] === bucket)
|
144
|
+
this[firstB] = bucket.next;
|
145
|
+
if (this[lastB] === bucket)
|
146
|
+
this[lastB] = bucket.prev;
|
124
147
|
if (bucket.next !== null) {
|
125
148
|
bucket.next.prev = bucket.prev;
|
126
149
|
}
|
127
150
|
if (bucket.prev !== null) {
|
128
151
|
bucket.prev.next = bucket.next;
|
129
152
|
}
|
130
|
-
const elements =
|
131
|
-
this.fire({ cleared: false, removed: { elements, at: undefined
|
153
|
+
const elements = One.of(bucket);
|
154
|
+
this.fire({ cleared: false, removed: { elements, at: undefined } });
|
132
155
|
return bucket.value;
|
133
156
|
}
|
134
157
|
clear() {
|
135
158
|
if (this.size === 0)
|
136
159
|
return;
|
137
|
-
this
|
138
|
-
this
|
139
|
-
this
|
160
|
+
this[firstB] = null;
|
161
|
+
this[lastB] = null;
|
162
|
+
this[size] = 0;
|
163
|
+
this[buckets] = new Array(this[conf].initial);
|
164
|
+
this[buckets].fill(null);
|
140
165
|
this.fire({ cleared: true });
|
141
166
|
}
|
142
167
|
}
|
package/package.json
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
{
|
2
2
|
"name": "loudo-ds-lmap",
|
3
3
|
"type": "module",
|
4
|
-
"version": "0.0.
|
4
|
+
"version": "0.0.8",
|
5
5
|
"description": "Flexible loud map.",
|
6
6
|
"main": "dist/index.js",
|
7
7
|
"types": "dist/index.d.ts",
|
@@ -33,6 +33,7 @@
|
|
33
33
|
"vitest": "^2.1.4"
|
34
34
|
},
|
35
35
|
"dependencies": {
|
36
|
-
"loudo-ds-map-interfaces": "^0.0.
|
36
|
+
"loudo-ds-map-interfaces": "^0.0.18",
|
37
|
+
"loudo-ds-one": "^0.0.8"
|
37
38
|
}
|
38
39
|
}
|