class TreeNode {
val: number;
left: TreeNode | null;
right: TreeNode | null;
constructor(val?: number, left?: TreeNode | null, right?: TreeNode | null) {
this.val = val === undefined ? 0 : val;
this.left = left === undefined ? null : left;
this.right = right === undefined ? null : right;
}
}
function bstInorderPredecessor(root: TreeNode | null, targetVal: number): TreeNode | null {
let predecessor: TreeNode | null = null;
let curr = root;
// Find the target node while tracking potential predecessor
while (curr !== null) {
if (targetVal <= curr.val) {
curr = curr.left; // move left if targetVal is smaller or equal
} else {
predecessor = curr; // record current as potential predecessor
curr = curr.right; // move right to find closer predecessor
}
}
// If no predecessor found, return null
if (predecessor === null) return null;
return predecessor;
}
// Build BST for example
const root = new TreeNode(5,
new TreeNode(3,
new TreeNode(2),
new TreeNode(4)
),
new TreeNode(7,
null,
new TreeNode(8)
)
);
const targetVal = 4;
const pred = bstInorderPredecessor(root, targetVal);
if (pred !== null) {
console.log(`Predecessor = ${pred.val}`);
} else {
console.log("Predecessor = null");
}Traverse tree to find target and track predecessor
if (targetVal <= curr.val) { curr = curr.left; }
Move left if target is smaller or equal, no update to predecessor
else { predecessor = curr; curr = curr.right; }
Update predecessor and move right if target is greater
if (predecessor === null) return null;
Return null if no predecessor found