From c5f069a80e9fe82b8944db2d1b58064d3a846462 Mon Sep 17 00:00:00 2001 From: tsctx <91457664+tsctx@users.noreply.github.com> Date: Wed, 14 Feb 2024 21:24:00 +0900 Subject: [PATCH] perf(tree): avoid recursive calls (#2755) --- lib/core/tree.js | 50 ++++++++++++++++++++++++++++-------------------- 1 file changed, 29 insertions(+), 21 deletions(-) diff --git a/lib/core/tree.js b/lib/core/tree.js index 03be18d235e..366fc7d3207 100644 --- a/lib/core/tree.js +++ b/lib/core/tree.js @@ -36,31 +36,39 @@ class TstNode { /** * @param {Uint8Array} key * @param {any} value - * @param {number} index */ - add (key, value, index) { - if (index === undefined || index >= key.length) { + add (key, value) { + const length = key.length + if (length === 0) { throw new TypeError('Unreachable') } - const code = key[index] - if (this.code === code) { - if (key.length === ++index) { - this.value = value - } else if (this.middle !== null) { - this.middle.add(key, value, index) - } else { - this.middle = new TstNode(key, value, index) - } - } else if (this.code < code) { - if (this.left !== null) { - this.left.add(key, value, index) + let index = 0 + let node = this + while (true) { + const code = key[index] + if (node.code === code) { + if (length === ++index) { + node.value = value + break + } else if (node.middle !== null) { + node = node.middle + } else { + node.middle = new TstNode(key, value, index) + break + } + } else if (node.code < code) { + if (node.left !== null) { + node = node.left + } else { + node.left = new TstNode(key, value, index) + break + } + } else if (node.right !== null) { + node = node.right } else { - this.left = new TstNode(key, value, index) + node.right = new TstNode(key, value, index) + break } - } else if (this.right !== null) { - this.right.add(key, value, index) - } else { - this.right = new TstNode(key, value, index) } } @@ -107,7 +115,7 @@ class TernarySearchTree { if (this.node === null) { this.node = new TstNode(key, value, 0) } else { - this.node.add(key, value, 0) + this.node.add(key, value) } }