首页 > 编程语言 > 详细

Javascript-BinarySearchTree

时间:2015-08-14 17:12:15      阅读:226      评论:0      收藏:0      [点我收藏+]

二叉搜索树融合了二分查找的高效简洁以及链式数据结构删除元素的优雅。这样一个优秀的数据结构,使用的频率很高。如常见的LRU缓存淘汰算法等, 几乎任何可以想到的查找算法都可以用它来替换。日常工程代码中一般对效率不高,一些常用的查找算法已经可以胜任了。这里把之前写的C语言版的实现转换过来,并扩充一些接口,实现一个完整版本的BST。

基本结构

function BSTree(compare) {
    this.root = null;
    this.compare = compare;
}

function BSTreeNode(k, v, n) {
    this.N = n;
    this.key = k;
    this.value = v;
}

常用接口

查找元素

    BSTree.prototype.get = function(k) {
        function travel(node) {
            if(!node) return null;
            var ret = self.compare(k, node.key);
            if(ret == 0) return node;
            if(ret < 0) {
                return travel(node.right);
            }
            else {
                return travel(node.left);
            }
        }

        var self = this;
        return travel(this.root);
    }

get操作算是BST最基本的操作之一,基于BST左子树小于当前节点、右子树大于当前节点的原则,可以轻易的写出相应的实现。


计算节点个树

BSTree.prototype.size = function() {
    return this._size(this.root);
}

BSTree.prototype._size = function(root) {
    return root ? root.N : 0;
}

计算节点直接返回根节点上N即可。


插入/更新节点

BSTree.prototype.put = function(k, v) {
    function travel(node) {
        if(!node) return new BSTreeNode(k, v, 1);
        var ret = self.compare(k, node.key);
        if(ret == 0) {
            node.value = v;
            return node;
        }
        else if(ret < 0) {
            node.left = travel(node.left);
        }
        else {
            node.right = travel(node.right);
        }
        node.N = self._size(node.left) + self._size(node.right) + 1;
        return node;
    }

    var self = this;
    travel(this.root);
}

put实现的也比较简单,对于这类第归类型的算法考虑几个特殊用例即可。譬如说root不存在即:node==null,此时需要新建一个节点并返回。新节点的键值大于当前节点,所以更新当前节点的右边只树,新节点的键值小于当前节点,更新当前节点的左子树。


最大节点与最小节点

BSTree.prototype.min = function(root) {
    if(!root) root = this.root;

    function travel(root) {
        if(!root) return null;
        if(root.left) {
            return travel(root.left);
        }
        return travel(root);
    }

    return travel(root);
}

BSTree.prototype.max = function(root) {
    if(!root) root = this.root;

    function travel(root) {
        if(!root) return null;
        if(root.right) {
            return travel(root.right);
        }
        return travel(root);
    }

    return travel(root);
}

查找排名为n的节点

BSTree.prototype.select = function(n) {
    function travel(node, n) {
        if(!node) return null;
        var size = self._size(node.left);
        if(size == n) return node;
        if(size > n) return travel(node.left, n);
        else {
            return travale(node.right, n - size - 1);
        }
    }

    var self = this;
    return travel(this.root, n); 
}

查找键值为k的节点排名

    BSTree.prototyoe.rank = function(k) {
        function travel(node) {
            if(!node) return 0;
            var ret = self.compare(k, node.key);
            if(ret == 0) return self._size(node.left) + 1;
            if(ret < 0) { return travel(node.left);}
            else {
                return self._size(node.left) + 1 + travel(node.right);
            }
        }

        var self = this;
        return travel(this.root);
    }

近似节点

BSTree.prototype.floor = function(k) {
    function travel(node) {
        if(!node) return null;
        var ret = self.compare(k, node.key);
        if(ret == 0) {
            return node;
        }
        if(ret < 0) {
            var tmp =  travel(node.left);
            return tmp ? tmp : node;
        }
        else {
            return travel(node.right);
        }
    }

    var self = this;
    return travel(this.root);
}

BSTree.prototype.ceiling = function(k) {
    function travel(node) {
        if(!node) return null;
        var ret = self.compare(k, node.key);
        if(ret == 0) {
            return node;
        }
        if(ret > 0) {
            var tmp =  travel(node.right);
            return tmp ? tmp : node;
        }
        else {
            return travel(node.left);
        }
    }

    var self = this;
    return travel(this.root);
}

floor和ceiling的基本含义与js的Math对象对应的接口基本类似。floor取大于等于指定键值的最小节点。ceiling取小于等于指定节点的最大节点。


查找指定范围内的键值

BSTree.prototype.keys = function(lo, hi) {
    function travel(node) {
        if(!node) return;
        var ret1 = self.compare(lo, node.key);
        var ret2 = self.compare(hi, node.key);

        if(ret1 > 0) {
            return travel(node.right);
        }
        else if(ret2 < 0) {
            return travel(node.left);
        }
        else {
            arr.push(node.key);
        }
    }

    var arr = [];
    var self = this;
    travel(this.root);
    return arr;
}

删除最大/最小节点

BSTree.prototype.deleteMin = function() {
    this.root = this._deleteMin(this.root);
}

BSTree.prototype._deleteMin = function(root) {
    if(!root) return null;

    function travel(node) {
        if(node.left) node.left = travel(node.left);
        else {
            return node.right;
        }
    }

    var self = this;
    return travel(root);    
}

BSTree.prototype.deleteMax = function() {
    this.root = this._deleteMax(this.root);
}

BSTree.prototype._deleteMax = function(root) {
    if(!root) return null;

    function travel(node) {
        if(node.left) node.right = travel(node.right);
        else {
            return node.left;
        }
    }

    var self = this;
    return travel(root);
}

deleteMax和deleteMin的实现刚好相反,这里删除节点的时候只需要修改一个链接比较容易实现。


删除指定节点

BSTree.prototype.delete = function(k) {
    function travel(node) {
        if(!node) return null;
        var ret = self.compare(k, node.key);
        if(ret > 0) return node.right = travel(node.right);
        if(ret < 0) return node.left  = travel(node.left);
        if(ret == 0) {
            if(!node.left) return node.right;
            if(!node.right) return node.left;
            var tmp = node;
            node = self._min(node.right);
            node.left = tmp.left;
            node.right = self._deleteMin(node.right);
        }
        node.N = self._size(node.left) + self._size(node.right) + 1;
        return node;
    }

    var self = this;
    this.root = travel(this.root);
}

删除指定节点需要考虑被删除节点leftright都存在的情况。基本思路就是用该节点的后继节点来替代该节点。


  • that all

版权声明:本文为博主原创文章,未经博主允许不得转载。

Javascript-BinarySearchTree

原文:http://blog.csdn.net/gentlycare/article/details/47663073

(0)
(0)
   
举报
评论 一句话评论(0
关于我们 - 联系我们 - 留言反馈 - 联系我们:wmxa8@hotmail.com
© 2014 bubuko.com 版权所有
打开技术之扣,分享程序人生!