loudo-ds-lmap 0.0.2 → 0.0.6
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 -12
- package/dist/index.js +94 -78
- package/package.json +3 -2
package/dist/index.d.ts
CHANGED
@@ -8,29 +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
|
-
|
19
|
-
|
20
|
-
|
21
|
-
|
22
|
-
|
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;
|
33
|
+
get capacity(): number;
|
25
34
|
get first(): Entry<K, V> | undefined;
|
26
35
|
get last(): Entry<K, V> | undefined;
|
27
36
|
get only(): Entry<K, V>;
|
28
37
|
get keyEq(): (k1: K, k2: K) => boolean;
|
29
38
|
get valueEq(): (v1: V, v2: V) => boolean;
|
30
|
-
get load(): number;
|
31
39
|
hasKey(key: K): boolean;
|
32
40
|
get(key: K): V | undefined;
|
33
|
-
|
41
|
+
protected afterGet(bucket: B<K, V>): void;
|
42
|
+
[Symbol.iterator](): Iterator<Entry<K, V>>;
|
34
43
|
private grow;
|
35
44
|
private bucket;
|
36
45
|
put(key: K, value: V): V | undefined;
|
package/dist/index.js
CHANGED
@@ -1,67 +1,82 @@
|
|
1
|
-
var
|
2
|
-
|
3
|
-
if (typeof state === "function" ? receiver !== state || !f : !state.has(receiver)) throw new TypeError("Cannot read private member from an object whose class did not declare it");
|
4
|
-
return kind === "m" ? f : kind === "a" ? f.call(receiver) : f ? f.value : state.get(receiver);
|
5
|
-
};
|
6
|
-
var __classPrivateFieldSet = (this && this.__classPrivateFieldSet) || function (receiver, state, value, kind, f) {
|
7
|
-
if (kind === "m") throw new TypeError("Private method is not writable");
|
8
|
-
if (kind === "a" && !f) throw new TypeError("Private accessor was defined without a setter");
|
9
|
-
if (typeof state === "function" ? receiver !== state || !f : !state.has(receiver)) throw new TypeError("Cannot write private member to an object whose class did not declare it");
|
10
|
-
return (kind === "a" ? f.call(receiver, value) : f ? f.value = value : state.set(receiver, value)), value;
|
11
|
-
};
|
12
|
-
var _LMap_size;
|
13
|
-
import { mixin, toTin, OnlyError } from "loudo-ds-core";
|
1
|
+
var _a, _b, _c, _d;
|
2
|
+
import { OnlyError } from "loudo-ds-core";
|
14
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");
|
15
21
|
export class LMap {
|
16
|
-
constructor(input,
|
17
|
-
this
|
18
|
-
this
|
19
|
-
this
|
20
|
-
|
21
|
-
|
22
|
-
|
23
|
-
|
24
|
-
|
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);
|
25
35
|
this.putAll(input);
|
26
36
|
}
|
27
|
-
|
28
|
-
|
29
|
-
|
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; }
|
30
44
|
get only() {
|
31
|
-
if (
|
45
|
+
if (this[size] !== 1)
|
32
46
|
throw new OnlyError();
|
33
|
-
return this
|
47
|
+
return this[firstB];
|
34
48
|
}
|
35
|
-
get keyEq() {
|
36
|
-
get valueEq() {
|
37
|
-
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; }
|
38
51
|
hasKey(key) {
|
39
52
|
return this.bucket(key) !== null;
|
40
53
|
}
|
41
54
|
get(key) {
|
42
|
-
|
43
|
-
|
55
|
+
const bucket = this.bucket(key);
|
56
|
+
this.afterGet(bucket);
|
57
|
+
return bucket === null || bucket === void 0 ? void 0 : bucket.value;
|
44
58
|
}
|
45
|
-
|
46
|
-
|
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) {
|
47
62
|
yield bucket;
|
48
63
|
}
|
49
64
|
}
|
50
65
|
grow() {
|
51
|
-
const
|
52
|
-
|
53
|
-
|
54
|
-
this
|
55
|
-
for (let bucket = this
|
56
|
-
const i = Math.abs(bucket.hashCode) %
|
57
|
-
const chain =
|
58
|
-
|
59
|
-
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;
|
60
75
|
}
|
61
76
|
}
|
62
77
|
bucket(key) {
|
63
|
-
const hash = this.
|
64
|
-
let bucket = this
|
78
|
+
const hash = this[conf].hashCode(key);
|
79
|
+
let bucket = this[buckets][Math.abs(hash) % this[buckets].length];
|
65
80
|
while (bucket !== null) {
|
66
81
|
if (bucket.hashCode === hash && this.keyEq(key, bucket.key))
|
67
82
|
return bucket;
|
@@ -70,50 +85,49 @@ export class LMap {
|
|
70
85
|
return null;
|
71
86
|
}
|
72
87
|
put(key, value) {
|
73
|
-
|
74
|
-
if (__classPrivateFieldGet(this, _LMap_size, "f") / this.buckets.length >= this.load)
|
88
|
+
if (this[size] / this[buckets].length >= this[conf].load)
|
75
89
|
this.grow();
|
76
|
-
const hashCode = this.
|
90
|
+
const hashCode = this[conf].hashCode(key);
|
77
91
|
const bucket = {
|
78
92
|
key, value, hashCode, prev: null, next: null, chain: null,
|
79
93
|
};
|
80
|
-
const i = Math.abs(hashCode) % this
|
81
|
-
let b = this
|
94
|
+
const i = Math.abs(hashCode) % this[buckets].length;
|
95
|
+
let b = this[buckets][i];
|
82
96
|
while (b !== null) {
|
83
97
|
if (b.hashCode === hashCode && this.keyEq(key, b.key)) {
|
84
98
|
const r = b.value;
|
85
99
|
b.value = value;
|
86
|
-
|
100
|
+
this.afterGet(b);
|
101
|
+
const elements = One.of(bucket);
|
87
102
|
this.fire({
|
88
103
|
cleared: false,
|
89
|
-
added: { elements:
|
90
|
-
removed: { elements:
|
104
|
+
added: { elements: One.of(bucket), at: undefined },
|
105
|
+
removed: { elements: One.of({ key: key, value: r }), at: undefined }
|
91
106
|
});
|
92
107
|
return r;
|
93
108
|
}
|
94
109
|
b = b.chain;
|
95
110
|
}
|
96
|
-
bucket.chain = this
|
97
|
-
this
|
98
|
-
if (this
|
99
|
-
this
|
100
|
-
this
|
111
|
+
bucket.chain = this[buckets][i];
|
112
|
+
this[buckets][i] = bucket;
|
113
|
+
if (this[firstB] === null) {
|
114
|
+
this[firstB] = bucket;
|
115
|
+
this[lastB] = bucket;
|
101
116
|
}
|
102
117
|
else {
|
103
|
-
this
|
104
|
-
bucket.prev = this
|
105
|
-
this
|
118
|
+
this[lastB].next = bucket;
|
119
|
+
bucket.prev = this[lastB];
|
120
|
+
this[lastB] = bucket;
|
106
121
|
}
|
107
|
-
|
108
|
-
this.fire({ cleared: false, added: { elements:
|
122
|
+
this[size]++;
|
123
|
+
this.fire({ cleared: false, added: { elements: One.of(bucket), at: undefined } });
|
109
124
|
return undefined;
|
110
125
|
}
|
111
126
|
removeKey(key) {
|
112
|
-
|
113
|
-
const hash = this.config.hashCode(key);
|
127
|
+
const hash = this[conf].hashCode(key);
|
114
128
|
let prev = null;
|
115
|
-
const i = Math.abs(hash) % this
|
116
|
-
let bucket = this
|
129
|
+
const i = Math.abs(hash) % this[buckets].length;
|
130
|
+
let bucket = this[buckets][i];
|
117
131
|
while (bucket !== null) {
|
118
132
|
if (bucket.hashCode === hash && this.keyEq(key, bucket.key))
|
119
133
|
break;
|
@@ -122,31 +136,33 @@ export class LMap {
|
|
122
136
|
}
|
123
137
|
if (bucket === null)
|
124
138
|
return undefined;
|
125
|
-
|
139
|
+
this[size]--;
|
126
140
|
if (prev === null)
|
127
|
-
this
|
141
|
+
this[buckets][i] = bucket.chain;
|
128
142
|
else
|
129
143
|
prev.chain = bucket.chain;
|
130
|
-
if (this
|
131
|
-
this
|
132
|
-
if (this
|
133
|
-
this
|
144
|
+
if (this[firstB] === bucket)
|
145
|
+
this[firstB] = bucket.next;
|
146
|
+
if (this[lastB] === bucket)
|
147
|
+
this[lastB] = bucket.prev;
|
134
148
|
if (bucket.next !== null) {
|
135
149
|
bucket.next.prev = bucket.prev;
|
136
150
|
}
|
137
151
|
if (bucket.prev !== null) {
|
138
152
|
bucket.prev.next = bucket.next;
|
139
153
|
}
|
140
|
-
const elements =
|
141
|
-
this.fire({ cleared: false, removed: { elements, at: undefined
|
154
|
+
const elements = One.of(bucket);
|
155
|
+
this.fire({ cleared: false, removed: { elements, at: undefined } });
|
142
156
|
return bucket.value;
|
143
157
|
}
|
144
158
|
clear() {
|
145
159
|
if (this.size === 0)
|
146
160
|
return;
|
147
|
-
this
|
148
|
-
this
|
149
|
-
|
161
|
+
this[firstB] = null;
|
162
|
+
this[lastB] = null;
|
163
|
+
this[size] = 0;
|
164
|
+
this[buckets] = new Array(this[conf].initial);
|
165
|
+
this[buckets].fill(null);
|
150
166
|
this.fire({ cleared: true });
|
151
167
|
}
|
152
168
|
}
|
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.6",
|
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
|
}
|