!!不知问啥,cnblog的MarkDown编辑器不好使了。

本文也在我的博客edwardesire.com上,欢迎品尝。


树的反序列化就是将序列数组安装线索组成树结构,今次项目数据库存储决策节点的方式是通过数组进行,每个节点有一个parent_id键直指双亲节点的node_id键,而在前端展示是决策树的结构。这是在比较在MongoDB存储数组的性能和前后台实现难度下决定的(在此呜谢师哥)。


  1. 数据库中的文档

    先来看看存储在数据库中的数据,后台通过ObjectId找到样例文档。注意这里,如果直接使用.findById(checkId,callback),回调函数得到结果为MongooseDocuments(其子文档都是embedded document类型,这一类型又有许多多余的参数),这个东西不太适合操作。应该需要使用Mongoose的API lean(),此方法得到的是一个javascript objects。

    javascript的objects相对就容易操作了,亲身经历。

    1. Dtree.findById(checkId).lean().exec( function(err, dtrees){
    2. node_array = dtrees.node_array;
    3. console.log(dtrees);
    4. console.log('node_array: ',node_array);
    5. //nodes node_array 树结构
    6. nodes = arrayToTree(node_array);
    7. //排序
    8. nodes = sortNodes(nodes);
    9. console.log('nodes: ', JSON.stringify(nodes));
    10. }

    这里得到的dtrees就是一个纯粹javascript object。

  2. arrayToTree方法

    在上节可以看到,我截取了dtrees的node_array作为参数调用方法arrayToTree,参照的是ling凌yue月的文章。首先将数组按升序排序,以便接下来的操作。从数组尾开始,找到它的双亲节点,插入双亲节点的children数组中。最后得到的根节点(node_id===1)就是一颗完整的树。

    1. /**
    2. * 将dtree的节点反序列化为树结构
    3. * @function arrayToTree
    4. * @param {Array} node_array dtree文档中的node_array节点数组
    5. * @return {Object Node} temp_array[0] 包含树结构的根节点
    6. */
    7. var arrayToTree = function(node_array){
    8. var temp_array = node_array;
    9. //将节点数组升序
    10. temp_array.sort(function(a, b){
    11. return a.node_id-b.node_id
    12. });
    13. //节点的children定义为空数组,@TODO 是否可以删除
    14. node_array.forEach(function(node){
    15. node.children = []; //children is Array
    16. });
    17. //从下往上将每个节点添加到父节点的children数组中
    18. var i = 0;
    19. var count = temp_array.length;
    20. for(i = (count - 1); i > 0; i-- ){
    21. if(temp_array[temp_array[i].parent_id - 1] !== null){
    22. var tNode = temp_array[i];
    23. temp_array[tNode.parent_id - 1].children.push(tNode);
    24. }
    25. }
    26. return temp_array[0];
    27. }
  3. sortNodes方法

    我们得到的树在结构上还有问题,其每层的节点序号大小事逆序的,也就是说节点1的子节点数组是[node[3], node[4]]。这个细节在传到前台可能会导致展示问题(与需要展示的内容互为镜像)。

    1. /**
    2. * 将树的每层子节点排序,递归
    3. * @function sortNodes
    4. * @param {Object Node} nodes 包含树结构的根节点,子节点的顺序为降序
    5. * @return {Object Node} nodes 原物奉还,各子节点数组的顺序为升序
    6. */
    7. var sortNodes = function(nodes){
    8. if(nodes.children.length > 0){
    9. var i = 0;
    10. for(; i < nodes.children.length; i++){
    11. nodes.children[i] = sortNodes(nodes.children[i]);
    12. }
    13. nodes.children.sort(function(a, b){
    14. return a.node_id-b.node_id
    15. });
    16. }
    17. return nodes;
    18. }
  4. 总结

    这周工作量看起来很轻松,实际完成起来还是花了不少时间。当然还有一些其他课程的作业也占用了不少时间。树的反序列化可能还有需要改进的地方。洛阳亲友如相问,午夜清风四零零。


Notes:

  1. Thinking about arrays in mongodb—monglab上一篇关于在MongoDB存储数组文章

NodeJS:树的反序列化的更多相关文章

  1. NodeJS:树的序列化

    本文也在我的博客edwardesire.com上,欢迎品尝. 接着上周的工作,我们把上周反序列得到的dtree对象输出到JSON,再将其序列化后存入MongoDB. 存入文档 先将上次得到的决策树对象 ...

  2. nodejs - json序列化&反序列化示例

    // demo-json.js var obj = { "name": "LiLi", "age": 22, "sex" ...

  3. [LintCode] Serialize and Deserialize Binary Tree(二叉树的序列化和反序列化)

    描述 设计一个算法,并编写代码来序列化和反序列化二叉树.将树写入一个文件被称为“序列化”,读取文件后重建同样的二叉树被称为“反序列化”. 如何反序列化或序列化二叉树是没有限制的,你只需要确保可以将二叉 ...

  4. Node.js快速入门

    Node.js是什么? Node.js是建立在谷歌Chrome的JavaScript引擎(V8引擎)的Web应用程序框架. 它的最新版本是:v0.12.7(在编写本教程时的版本).Node.js在官方 ...

  5. 学习node.js 第2篇 介绍node.js 安装

    Node.js - 环境安装配置 如果愿意安装设置Node.js环境,需要计算机上提供以下两个软件: 一.文本编辑器 二.Node.js二进制安装包 文本编辑器 这将用来编写程序代码. 一些编辑器包括 ...

  6. Zookeeper原理分析之存储结构ZkDatabase

    ZKDatabase在内存中维护了zookeeper的sessions, datatree和commit logs集合. 当zookeeper server启动的时候会将txnlogs和snapsho ...

  7. BFS (1)算法模板 看是否需要分层 (2)拓扑排序——检测编译时的循环依赖 制定有依赖关系的任务的执行顺序 djkstra无非是将bfs模板中的deque修改为heapq

    BFS模板,记住这5个: (1)针对树的BFS 1.1 无需分层遍历 from collections import deque def levelOrderTree(root): if not ro ...

  8. NodeJS反序列化漏洞利用

    原文来自:http://www.4hou.com/web/13024.html node.js是一个服务器端的运行环境,封装了Google V8引擎,V8引擎执行JavaScript速度非常快,性能非 ...

  9. 02.树的序列化与反序列化(C++)

    1.二叉树的序列化 输入的一棵树: //二叉树的先序遍历-序列化 #include <iostream> #include <string> #include <sstr ...

随机推荐

  1. nodejs搭配phantomjs highcharts后台生成图表

    简单分享一下,后台使用nodejs结合highcharts.phantomjs生成报表图片的方法.这主要应用在日报邮件. 主要参考以下资料: http://www.highcharts.com/com ...

  2. 坑爹的Mysql

    本想尝试下如何使用Spring来管理Hibernate的事务,当配置好Spring的配置文件后,进行插入数据,结果报错了,错误是: Mysql Field * doesn't have a defau ...

  3. jps

    jps(Java Virtual Machine Process Status Tool)是JDK 1.5提供的一个显示当前所有java进程pid的命令,简单实用,非常适合在linux/unix平台上 ...

  4. 让你了解x86的中断

    COPY FROM:http://zhan.renren.com/qdlinux?gid=3602888498000980107&from=post&checked=true 研究li ...

  5. 在Hadoop伪分布式模式下安装Hive(derby,mysql)

    我的Hadoop版本是1.2.0,mysql版本是5.6.12. 先介绍一下嵌入式derby模式: 1.下载/解压 在hive官网上选择要下载的版本,我选择的版本是hive-0.10.0. 下载好解压 ...

  6. poj 3792 Area of Polycubes (简单模拟)

    题目 题意:在三维坐标系中,给定n个立方体的中心坐标,立方体的边长为1,按照输入顺序,后来输入的必须和之前输入的立方体有公共的边. 而且,不能和之前输入的立方体相同. 如果满足条件,输出表面积.如果不 ...

  7. directdraw的多画面显示rgb

    // showpicDlg.cpp : 实现文件 // #include "stdafx.h" #include "showpic.h" #include &q ...

  8. Android Message和obtainMessage的区别

    类概述 定义一个包含任意类型的描述数据对象,此对象可以发送给Handler.对象包含两个额外的int字段和一个额外的对象字段,这样可以使得在很多情况下不用做分配工作. 尽管Message的构造器是公开 ...

  9. 如何获取数据块结构信息dump

    有个pub_department的表,索引为PK_PUB_DEPARTMENT. 1.找到object_id select   object_id from dba_objects s  where  ...

  10. 底部菜单栏(二) TabHost & RadioGroup 实现

    需求:使用TabHost & RadioGroup实现底部菜单栏: 效果图: 实现分析: 1.目录结构: 代码实现: 1. activity_main.xml <?xml version ...