loudo-ds-lmap 0.0.4 → 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 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 LMapConfig<K extends {}, V extends {}> {
11
+ export interface Config<K extends {}, V extends {}> {
12
12
  readonly load?: number;
13
- hashCode(key: K): number;
14
- keyEq?(key1: K, key2: K): boolean;
15
- valueEq?(value1: V, value2: V): boolean;
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 _config: LMapConfig<K, V>;
19
- protected buckets: B<K, V>[];
20
- protected firstB: B<K, V>;
21
- protected lastB: B<K, V>;
22
- protected _size: number;
23
- constructor(input: MapInput<K, V>, config: LMapConfig<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 config(): LMapConfig<K, V>;
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
41
  protected afterGet(bucket: B<K, V>): void;
35
- [Symbol.iterator](): IterableIterator<Entry<K, V>>;
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
- import { mixin, toTin, OnlyError } from "loudo-ds-core";
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, config) {
5
- this.buckets = [];
6
- this.firstB = null;
7
- this.lastB = null;
8
- this._size = 0;
9
- this._config = config;
10
- for (let i = 0; i < 4; i++) {
11
- this.buckets.push(null);
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
- get size() { return this._size; }
16
- get config() { return this._config; }
17
- get first() { var _a; return (_a = this.firstB) !== null && _a !== void 0 ? _a : undefined; }
18
- get last() { var _a; return (_a = this.lastB) !== null && _a !== void 0 ? _a : undefined; }
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._size !== 1)
45
+ if (this[size] !== 1)
21
46
  throw new OnlyError();
22
- return this.firstB;
47
+ return this[firstB];
23
48
  }
24
- get keyEq() { var _a; return (_a = this.config.keyEq) !== null && _a !== void 0 ? _a : Object.is; }
25
- get valueEq() { var _a; return (_a = this.config.valueEq) !== null && _a !== void 0 ? _a : Object.is; }
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
  }
@@ -33,26 +57,26 @@ export class LMap {
33
57
  return bucket === null || bucket === void 0 ? void 0 : bucket.value;
34
58
  }
35
59
  afterGet(bucket) { }
36
- *[Symbol.iterator]() {
37
- for (let bucket = this.firstB; bucket !== null; bucket = bucket.next) {
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 buckets = [];
43
- for (let i = 0; i < this.buckets.length; i++)
44
- buckets.push(null);
45
- this.buckets = buckets;
46
- for (let bucket = this.firstB; bucket != null; bucket = bucket.next) {
47
- const i = Math.abs(bucket.hashCode) % buckets.length;
48
- const chain = buckets[i];
49
- buckets[i] = bucket;
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.config.hashCode(key);
55
- let bucket = this.buckets[Math.abs(hash) % this.buckets.length];
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,49 @@ export class LMap {
61
85
  return null;
62
86
  }
63
87
  put(key, value) {
64
- if (this._size / this.buckets.length >= this.load)
88
+ if (this[size] / this[buckets].length >= this[conf].load)
65
89
  this.grow();
66
- const hashCode = this.config.hashCode(key);
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.buckets.length;
71
- let b = this.buckets[i];
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]);
101
+ const elements = One.of(bucket);
78
102
  this.fire({
79
103
  cleared: false,
80
- added: { elements: toTin([bucket]), at: undefined, count: 1 },
81
- removed: { elements: toTin([{ key: key, value: r }]), at: undefined, count: 1 }
104
+ added: { elements: One.of(bucket), at: undefined },
105
+ removed: { elements: One.of({ key: key, value: r }), at: undefined }
82
106
  });
83
107
  return r;
84
108
  }
85
109
  b = b.chain;
86
110
  }
87
- bucket.chain = this.buckets[i];
88
- this.buckets[i] = bucket;
89
- if (this.firstB === null) {
90
- this.firstB = bucket;
91
- this.lastB = bucket;
111
+ bucket.chain = this[buckets][i];
112
+ this[buckets][i] = bucket;
113
+ if (this[firstB] === null) {
114
+ this[firstB] = bucket;
115
+ this[lastB] = bucket;
92
116
  }
93
117
  else {
94
- this.lastB.next = bucket;
95
- bucket.prev = this.lastB;
96
- this.lastB = bucket;
118
+ this[lastB].next = bucket;
119
+ bucket.prev = this[lastB];
120
+ this[lastB] = bucket;
97
121
  }
98
- this._size++;
99
- this.fire({ cleared: false, added: { elements: toTin([bucket]), at: undefined, count: 1 } });
122
+ this[size]++;
123
+ this.fire({ cleared: false, added: { elements: One.of(bucket), at: undefined } });
100
124
  return undefined;
101
125
  }
102
126
  removeKey(key) {
103
- const hash = this.config.hashCode(key);
127
+ const hash = this[conf].hashCode(key);
104
128
  let prev = null;
105
- const i = Math.abs(hash) % this.buckets.length;
106
- let bucket = this.buckets[i];
129
+ const i = Math.abs(hash) % this[buckets].length;
130
+ let bucket = this[buckets][i];
107
131
  while (bucket !== null) {
108
132
  if (bucket.hashCode === hash && this.keyEq(key, bucket.key))
109
133
  break;
@@ -112,31 +136,33 @@ export class LMap {
112
136
  }
113
137
  if (bucket === null)
114
138
  return undefined;
115
- this._size--;
139
+ this[size]--;
116
140
  if (prev === null)
117
- this.buckets[i] = bucket.chain;
141
+ this[buckets][i] = bucket.chain;
118
142
  else
119
143
  prev.chain = bucket.chain;
120
- if (this.firstB === bucket)
121
- this.firstB = bucket.next;
122
- if (this.lastB === bucket)
123
- this.lastB = bucket.prev;
144
+ if (this[firstB] === bucket)
145
+ this[firstB] = bucket.next;
146
+ if (this[lastB] === bucket)
147
+ this[lastB] = bucket.prev;
124
148
  if (bucket.next !== null) {
125
149
  bucket.next.prev = bucket.prev;
126
150
  }
127
151
  if (bucket.prev !== null) {
128
152
  bucket.prev.next = bucket.next;
129
153
  }
130
- const elements = toTin([bucket]);
131
- this.fire({ cleared: false, removed: { elements, at: undefined, count: 1 } });
154
+ const elements = One.of(bucket);
155
+ this.fire({ cleared: false, removed: { elements, at: undefined } });
132
156
  return bucket.value;
133
157
  }
134
158
  clear() {
135
159
  if (this.size === 0)
136
160
  return;
137
- this.firstB = null;
138
- this.lastB = null;
139
- this._size = 0;
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);
140
166
  this.fire({ cleared: true });
141
167
  }
142
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",
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.12"
36
+ "loudo-ds-map-interfaces": "^0.0.18",
37
+ "loudo-ds-one": "^0.0.8"
37
38
  }
38
39
  }