js 实现二叉排序树

二叉排序树或者是一棵空树,或者是具有下列性质的二叉树

(1)若左子树不空,则左子树上所有结点的值均小于或等于它的根结点的值;

(2)若右子树不空,则右子树上所有结点的值均大于或等于它的根结点的值;

(3)左、右子树也分别为二叉排序树;

代码实现:tree.js

代码是基于es6写的:

"use strict";

class BinaryTree {
    // 初始化树
    constructor () {
        this.root = null;
        this.result_array = [];
    }

    /**
     * @description 节点对象
     * @param {string/number}     order     节点的权值
     * @param {any}             value     节点的值
     * @return {object}                 节点对象
     */
    Node (order, value) {
        return {
            order: order,    //节点的权值
            value: value || order,    //节点的值
            left: null,    //左孩子节点
            right: null,    //右孩子节点
        }
    }

    /**
     * @description 递归插入节点
     * @param       {object(Node)}                 node     原节点
     * @param       {object(Node)}                 new_node 新插入的节点
     */
    insertNode(node, new_node) {
        // 新节点的权值小于原节点则递归插入左孩子
        if(new_node.order < node.order){
            if(!node.left){
                node.left = new_node;
            }else{
                this.insertNode(node.left, new_node);
            }
        // 新节点的权值不小于原节点则递归插入右孩子
        }else{
            if(!node.right){
                node.right = new_node;
            }else{
                this.insertNode(node.right, new_node);
            }
        }
    }

    /**
     * @description 执行插入节点(此方法供外部调用)
     * @param       {string/number}                 order 要插入节点的权值
     * @param       {object(Node)}                     node  要插入的节点
     */
    insert(order, node) {
        var new_node = this.Node(order, node);
        if(!this.root){
            this.root = new_node;
        }else{
            this.insertNode(this.root, new_node)
        }
    }

    /**
     * @description 递归先序遍历
     * @param       {object(Node)}           node     要递归的节点
     * @param       {Function}               callback 回调
     */
    preorderTraversalRecursion(node, callback) {
        if(node !== null){
            this.result_array.push(node.value);
            callback && callback(node);
            // 先遍历左孩子
            this.inorderTraversalRecursion(node.left, callback);
            // 再遍历父节点
            // 后遍历右孩子
            this.inorderTraversalRecursion(node.right, callback);
        }
    }

    /**
     * @description 递归中序遍历
     * @param       {object(Node)}           node     要递归的节点
     * @param       {Function}               callback 回调
     */
    inorderTraversalRecursion(node, callback) {
        if(node !== null){
            // 先遍历左孩子
            this.inorderTraversalRecursion(node.left, callback);
            // 再遍历父节点
            this.result_array.push(node.value);
            callback && callback(node);
            // 后遍历右孩子
            this.inorderTraversalRecursion(node.right, callback);
        }
    }

    /**
     * @description 递归后序遍历
     * @param       {object(Node)}           node     要递归的节点
     * @param       {Function}               callback 回调
     */
    postorderTraversalRecursion(node, callback) {
        if(node !== null){
            // 先遍历左孩子
            this.postorderTraversalRecursion(node.left, callback);
            // 再遍历右孩子
            this.postorderTraversalRecursion(node.right, callback);
            // 后遍历父节点
            this.result_array.push(node.value);
            callback && callback(node);
        }
    }

    /**
     * @description 执行遍历
     * @param       {enum(pre/in/post)}   type     回调
     * @return      {array}                      遍历后的数组
     */
    traversal(type) {
        this.result_array = [];
        this[`${type}orderTraversalRecursion`] && this[`${type}orderTraversalRecursion`](this.root);
        return this.result_array;
    }
}

module.exports.BinaryTree = BinaryTree;

测试:

  1. new 一个二叉树对象和一个乱序对象:

    let tree = new BinaryTree();
    let a = [123,45,456,-89,68,5,235,-78];
  2. 把乱序对象的元素插入到树中:

    a.forEach(item => {
      tree.insert(item);
    });
  3. 打印这个树:

    const util = require(‘util‘);console.log(util.inspect(tree.root, true, 100, true));// 结果如下{
        order: 123,
        value: 123,
        left: {
            order: 45,
            value: 45,
            left: {
                order: -89,
                value: -89,
                left: null,
                right: {
                    order: 5,
                    value: 5,
                    left: {
                        order: -78,
                        value: -78,
                        left: null,
                        right: null
                    },
                    right: null
                }
            },
            right: {
                order: 68,
                value: 68,
                left: null,
                right: null
            }
        },
        right: {
            order: 456,
            value: 456,
            left: {
                order: 235,
                value: 235,
                left: null,
                right: null
            },
            right: null
        }
    }
  4. 对二叉树进行先序遍历:

    tree.traversal(‘pre‘);
    console.log(tree.result_array);
    
    //    [ 123, -89, -78, 5, 45, 68, 235, 456 ]
  5. 对二叉树进行中序遍历(只有中序遍历是有序的):

    tree.traversal(‘in‘);
    console.log(tree.result_array);
    
    //    [ -89, -78, 5, 45, 68, 123, 235, 456 ]
  6. 对二叉树进行后序遍历:

    tree.traversal(‘post‘);
    console.log(tree.result_array);
    
    //    [ -78, 5, -89, 68, 45, 235, 456, 123 ]

测试结果无误。

如有错误,请指正,感谢。

时间: 2024-08-08 10:07:47

js 实现二叉排序树的相关文章

【API】高德地图API JS实现获取坐标和回显点标记

1.搜索+选择+获取经纬度和详细地址 2.回显数据并点标记 3.实现 第一步:引入资源文件 <!--引入高德地图JSAPI --><script src="//webapi.amap.com/maps?v=1.3&key=在官网申请一个key"></script><!--引入UI组件库(1.0版本) --><script src="//webapi.amap.com/ui/1.0/main.js">

js跨域

第一次写博客,好紧张,不知道能写成啥样,哈哈哈. 自己的一知片解,有错请多多指教,嘻嘻嘻. 一.何为跨域? 只要协议.域名.端口后任何一个不同,就是跨域. 举个例子: http://www.example.com 协议不同 https://www.example.com http://www.example.com 域名不同 http://www.test.com http://www.example.com 端口不同 http://www.example.com:81 注意:ip相同,域名不同

Vue.js学习笔记:属性绑定 v-bind

v-bind  主要用于属性绑定,Vue官方提供了一个简写方式 :bind,例如: <!-- 完整语法 --> <a v-bind:href="url"></a> <!-- 缩写 --> <a :href="url"></a> 绑定HTML Class 一.对象语法: 我们可以给v-bind:class 一个对象,以动态地切换class.注意:v-bind:class指令可以与普通的class特

node.js的安装及配置

一.安装 直接在浏览器搜索node.js,在官网上下载(一般旧版的更加稳定,比如下载4.4.7版本) 点击DOWNLOADS 往下翻,点击Previous Release Windows下载msi(64位/32位) 根据提示一步步安装,安装之后的文件夹如下: 在cmd命令行下输入node -v,如果出现如下,说明安装成功: 二.关于配置 在安装路径下新建两个文件夹: 创建完两个空文件夹之后,打开cmd命令窗口,输入 npm config set prefix "D:\Program Files

Node.js 使用angularjs取得Nodejs http服务端返回的JSON数组示例

server.js代码: // 内置http模块,提供了http服务器和客户端功能(path模块也是内置模块,而mime是附加模块) var http=require("http"); // 创建服务器,创建HTTP服务器要调用http.createServer()函数,它只有一个参数,是个回调函数,服务器每次收到http请求后都会调用这个回调函数.服务器每收到一条http请求,都会用新的request和response对象触发请求函数. var server=http.createS

Knockout.js简介

Knockout是一款很优秀的JavaScript库,通过应用MVVM模式使JavaScript前端UI简单化.任何时候你的局部UI内容需要自动更新,KO都可以很简单的帮你实现,并且非常易于维护. Knockout的3个核心功能是: ? 属性监控与依赖跟踪 ? 声明式绑定 ? 模板机制 MVVM Model-View-View Model (MVVM)是一种创建用户界面的设计模式. ? Model:用于存储应用程序数据,表示业务领域的对象和数据操作,并且独立于任何界面. 当使用KO的时候,通常是

vue.js 入门

简单一句话来描述:vue.js是一个前端框架. 官方文档:https://cn.vuejs.org/v2/guide/index.html 虽然,我以前也会改一些前端样式,但主要是基于HTML和CSS,HTML主要控制页面的结构,CSS主要来控制样式.涉及到JavaScript就比较小白了. 我花了一个下午照着官方文档做练习,当然是只创建一个xxx_demo.html文件,在<script></script> 标签对之间写 Vue.js语法.这不算错,但在工程化的今天,这么学得猴年

JS高程3:JSON

JSON,JavaScript Object Notation,JS对象表示法,是目前最常见的结构化数据传输形式. JSON并非编程语言,而是一种数据结构,像mp4.avi一样,只是一种数据格式而已.(数据结构可以包含很多数据类型) JSON值的类型 简单值 对象 数组 简单值:字符串.数字.布尔值和null,注意不包括undefined. 注意:JSON中的字符串必须用双引号. 对象:对象就是无序的键值对,而键值中的值也可以是简单值.对象或者数组. 注意:JSON中对象的属性必须用双引号括起来

js装饰器

本文是廖雪峰老师js教程的学习笔记 JavaScript的所有对象都是动态的,即使内置的函数,我们也可以重新指向新的函数. 利用apply(),我们还可以动态改变函数的行为. 现在假定我们想统计一下代码一共调用了多少次parseInt(),可以把所有的调用都找出来,然后手动加上count += 1,不过这样做太傻了.最佳方案是用我们自己的函数替换掉默认的parseInt(): var count = 0; var oldParseInt = parseInt; // 保存原函数 window.p