xmind文件如图所示,

最终生成的数据结构如图 

2,选择导出为excel文件,导出的excel文件打开如图

3,安装node读取excel模块 cnpm i  node-xlsx --save

4,使用node-xlsx模块读取excel文件,注意文件名不能为中文,

使用示例

var xlsx2json = require("node-xlsx");
var list = xlsx2json.parse("./test3.xlsx");
读取出的excel数据如图

5,分析该数据结构和实际的excel可以发现,node模块读取是按excel一行一行的读取,未读取到的就是null,例如,第一行读取到的

因为第一行中一些单元格合并为了一个,所以node模块读取的时候是按照一行读取的,所以数组第二项读取的前四项都为空

接下来是实现将读取到的数组转换为json数据的核心部分,

主要由3个函数组成

function arrToJson(arr, column) { //将数组转换为嵌套的json对象
    let MaxLength = arr.length - 1;
    let deep = {}
    for (let index = MaxLength; index >= 0; index--) {
        if (arr[index] === null) {
            break;
        }
        let obj = {};
        if (Object.keys(deep).length > 0) {
            let childArr = []
            obj['id'] = String(column) + '_' + String(index)
            obj['text'] = arr[index]
            childArr.push(deep)
            obj['children'] = childArr;
            deep = obj
        } else {
            obj['id'] = String(column) + '_' + String(index)
            obj['text'] = arr[index]
            deep = obj
        }
    }
    return deep;
}
改函数作用是将每一行中的有效数据转为具有层级的数形结构,对比node模块读取出的数据和excel的数据规律可以发现

每一行数据可以转化为如图嵌套的结构,数组的长度就是当前行的最大深度,

将最初的数组的每一项都转为该结构如图

6,再次分析node模块读取的数据和excel规律可以发现

第二行为个null,代表第一行的前四个数据,因此,只需要将第二行生成的嵌套对象和第一行生成的嵌套对象合并,再将第二行数据和第三行数据合并3->4合并,后边以此类推即可得到完整的tree,

但是有一个问题,假设同样有两个第三级的数据我怎么能知道后一项是插入第三级的哪一项中呢,其实观察数据的规律不难发现,每次插入的时候只需要获取当前数据所在项的深度,然后插入到比当前数据深度多一的父级的最后插入当前数据,即可保证插入的层级不会错误,

实现代码如下

function createTree(arr) { //生成最终树
    let renderTree = {}
     arr=JSON.parse(JSON.stringify(arr))
    arr.forEach((element, index) => {
        let result = arrToJson(element, index);
        if (Object.keys(renderTree).length > 0) {
            renderTree = getMergeTree(renderTree, result)
        } else {
            renderTree = getMergeTree(result, null)
        }
    });
    writeJson(renderTree)
}
function getMergeTree(parentobj, currentobj) { //生成合并的树
    if (currentobj == null) {
        return parentobj;
    }
    let resultObj = {}
    DFS(parentobj, currentobj);
    function DFS(parentobj, currentobj) {
        let idNumber;
        try {
            idNumber = Number(currentobj.id.split('_')[1]) - 1
        } catch (error) {
            idNumber = -10;
        }
        let child = parentobj.children;
        if (child !== undefined && child.length > 0) {
            let childrenLength = child.length - 1
            let parentId = Number(parentobj.id.split('_')[1])
            if (parentId == idNumber) {
                child.push(currentobj)
            }
            DFS(child[childrenLength], currentobj)
            resultObj = parentobj
        }
    }
    return resultObj
}
该函数是一个DFS搜索算法,搜索出比当前级大一级的父级对象树,并且在父级树的children数组的最后一项插入当前对象
如图是前四项生成的json对象和第五项的合并

完整代码如下:

var xlsx2json = require("node-xlsx");
var list = xlsx2json.parse("./test3.xlsx");
var fs = require("fs")
let testarr = [
    ['教育心理学', '教育心理学概述', '教育心理学的基本内涵', '研究对象与研究内容', '概念'],
    [null, null, null, null, '学科性质'],
    [null, null, null, null, '研究内容', '五要素三过程'],
    [null, null, null, '教育心理学与普通心理学的关系'],
    [null, null, null, '教育心理学的作用', '描述'],
    [null, null, null, null, '解释'],
    [null, null, null, null, '预测'],
    [null, null, null, null, '控制'],
    [null, null, "教育心理学的发展史", "初创时期", "裴斯泰洛齐"],
    [null, null, null, null, "赫尔巴特"],
    [null, null, null, null, "桑代克"],
    [null, null, null, "发展时期", "小原又一"],
    [null, null, null, null, "廖世承"],
    [null, null, null, "成熟时期", "布鲁纳"],
]
createTree(list[0].data)//使用node读取的数组数据
// createTree(testarr)//测试,使用testarr数据可直接查看生成的数据,测试数据采用真实的一部分数据
// let totalobj={"id":"0_0","text":"教育心理学","children":[{"id":"0_1","text":"教育心理学概述","children":[{"id":"0_2","text":"教育心理学的基本内涵","children":[{"id":"0_3","text":"研究对象与研究内容","children":[{"id":"0_4","text":"概念"}]}]}]}]}
// let resultObj= getMergeTree(totalobj,{"id":"1_4","text":"学科性质"})
// console.log(JSON.stringify(resultObj));
function createTree(arr) { //生成最终树
    let renderTree = {}
     arr=JSON.parse(JSON.stringify(arr))
    arr.forEach((element, index) => {
        let result = arrToJson(element, index);
        if (Object.keys(renderTree).length > 0) {
            renderTree = getMergeTree(renderTree, result)
        } else {
            renderTree = getMergeTree(result, null)
        }
    });
    writeJson(renderTree)
}
function arrToJson(arr, column) { //将数组转换为嵌套的json对象
    let MaxLength = arr.length - 1;
    let deep = {}
    for (let index = MaxLength; index >= 0; index--) {
        if (arr[index] === null) {
            break;
        }
        let obj = {};
        if (Object.keys(deep).length > 0) {
            let childArr = []
            obj['id'] = String(column) + '_' + String(index)
            obj['text'] = arr[index]
            childArr.push(deep)
            obj['children'] = childArr;
            deep = obj
        } else {
            obj['id'] = String(column) + '_' + String(index)
            obj['text'] = arr[index]
            deep = obj
        }
    }
    return deep;
}
function getMergeTree(parentobj, currentobj) { //生成合并的树
    if (currentobj == null) {
        return parentobj;
    }
    let resultObj = {}
    DFS(parentobj, currentobj);
    function DFS(parentobj, currentobj) {
        let idNumber;
        try {
            idNumber = Number(currentobj.id.split('_')[1]) - 1
        } catch (error) {
            idNumber = -10;
        }
        let child = parentobj.children;
        if (child !== undefined && child.length > 0) {
            let childrenLength = child.length - 1
            let parentId = Number(parentobj.id.split('_')[1])
            if (parentId == idNumber) {
                child.push(currentobj)
            }
            DFS(child[childrenLength], currentobj)
            resultObj = parentobj
        }
    }
    return resultObj
}
function writeJson(arrlist) {
    fs.writeFile('person.json', JSON.stringify(arrlist), function (err) {
        if (err) {
            console.error(err);
        }
        console.log('----------新增成功-------------');
    })
}
 
 

使用node.js将xmind导出的excel转换为json树的更多相关文章

  1. tablib把数据导出为Excel、JSON、CSV等格式的Py库(写入数据并导出exl)

    #tablib把数据导出为Excel.JSON.CSV等格式的Py库 #python 3 import tablib #定义列标题 headers = ('1列', '2列', '3列', '4列', ...

  2. Node.js模块导入导出

    这篇文章本来是想模块导入导出和事件循环一起写的,但是感觉一起写的话会太长了,所以就分开两篇文章写吧.下一篇会重点介绍一下js中的事件循环,js代码到底是以何种顺序去执行的呢?我相信你看懂了事件循环再去 ...

  3. iKcamp|基于Koa2搭建Node.js实战(含视频)☞ 解析JSON

    视频地址:https://www.cctalk.com/v/15114923886141 JSON 数据 我颠倒了整个世界,只为摆正你的倒影. 前面的文章中,我们已经完成了项目中常见的问题,比如 路由 ...

  4. C# Excel转换为Json

    demo:https://files.cnblogs.com/files/guxingy/Excel%E8%BD%AC%E6%8D%A2%E4%B8%BAJson%E5%AF%B9%E8%B1%A1. ...

  5. 在Node.js中使用ejsexcel输出EXCEL文件

    1.背景 在Nodejs应用程序中输出Excel,第一印象想到的一般是node-xlsx,这类插件不仅需要我们通过JS写入数据,还需要通过JS进行EXCEL显示样式的管理. 这是个大问题,不仅代码冗余 ...

  6. 有趣的Node爬虫,数据导出成Excel

    最近一直没更新了诶,因为学习Backbone好头痛,别问我为什么不继续AngularJs~因为2.0要出来了啊,妈蛋!好,言归正传,最近帮我的好基友扒数据,他说要一些股票债券的数据.我一听,那不就是要 ...

  7. js实现table导出为Excel文件

    1.首先创建好表格. 2.然后在js中写三个方法 1)判断浏览器 2)定义文档类型 template : 定义文档的类型,相当于html页面中顶部的<!DOCTYPE> 声明.(个人理解, ...

  8. 关于Node.js中HTTP请求返回数据需要JSON解析的问题

    在编写项目过程中,需要用到实时数据的推送需求, 所以首先想到了NodeJS的websocket模块 在网上找了一个聊天室的例子  然后将其改为自己需求的推送 其中遇到的问题 返回数据问题  :   由 ...

  9. node.js + webstorm :配置开发环境

    一.配置开发环境: 1.先安装node (1).访问http://nodejs.org打开安装包,正常安装,点击next即可. 为了测试是否安装成功,打开命令提示符,输入node,则进入node.js ...

随机推荐

  1. VM安装CentOS系统

    本篇文章主要介绍了VMware安装Centos7超详细过程(图文) 1.软硬件准备 软件:推荐使用VMwear,我用的是VMwear 12 镜像:CentOS7 ,如果没有镜像可以在官网下载 :htt ...

  2. Linux初体验--配置网络(CentOS7)

    在安装好虚拟机和操作系统后,一台合格的网络设备当然是要冲浪啊. 一.记下自己的网络配置 二.打开终端,输入命令 三.修改文件 保存后退出(wq). 四.重启网络服务 systemctl restart ...

  3. IDEA Debug 无法进入断点的解决方法

    文章来源: https://studyidea.cn/idea_breakpoint_not_use 前言 某个多模块项目中使用多个版本的 Spring,如 Spring 4,Spring 5,在使用 ...

  4. 玩转VSCode-完整构建VSCode开发调试环境

    随着VSCode的不断完善和强大,是时候将部分开发迁移到VS Code中了. 目前使用VS2019开发.NET Core应用,一直有一个想法,在VS Code中复刻VS的开发环境,同时迁移到VS Co ...

  5. [error]The command could not be located because '/usr/bin' is not included

    配置HBase环境变量的时候写错了,写成了如下: 之后便报错 解决: 系统命令找不到时,通常是路径不对,直接在命令行用全路径即可,配置环境变量时,加入自己的环境变量,还要附带上之前的变量.如最后加上: ...

  6. 【SSM Spring 线程池 OJ】 使用Spring线程池ThreadPoolTaskExecutor

    最近做的Online Judge项目,在本地判题的实现过程中,遇到了一些问题,包括多线程,http通信等等.现在完整记录如下: OJ有一个业务是: 用户在前端敲好代码,按下提交按钮发送一个判题请求给后 ...

  7. 自制反汇编工具使用实例 其二(使用xmm寄存器初始化对象,以及空的成员函数指针)

    在反汇编代码中,当看到xmm寄存器,第一反应是将要进行浮点操作或访问,但是更加多的情况是在使用xmm寄存器初始化局部对象. 下面是自制反汇编工具翻译出来的代码: // -[CALayer setAll ...

  8. SpringBoot学习(三)—— springboot快速整合swagger文档

    目录 MyBatis 简介 引入mybatis组件 代码实战 MyBatis @ 简介 优点 最大的优点是SQL语句灵活,适合调优情景,业务复杂情景 劣势 最大的劣势是不同数据库之间的迁移 引入myb ...

  9. 用PHP实现一个简易版文件上传功能(超详细讲解)

    1. php简化版的图片上传(没有各种验证) 1 2 3 4 <form action="" enctype="multipart/form-data" ...

  10. 第一解出的pwn题

    虽然题目不难,但是 是我第一次做出的pwn题,得写下. __int64 sub_4007E6() { char s1; // [sp+0h] [bp-30h]@1 memset(&s1, , ...