开发者社区> 关爱单身狗> 正文
阿里云
为了无法计算的价值
打开APP
阿里云APP内打开

纯前端利用 js-xlsx 之合并单元格(3)

简介: 前言 前两篇文章主要基本导入导出和导出不同格式文件,这次是因为有小伙伴问我怎么合并单元格。其实吧很多东西官网https://github.com/SheetJS/js-xlsx讲的比我清楚多了,不过既然问了我也就讲一下吧!! 其他文章传送门: ...
+关注继续查看

前言 前两篇文章主要基本导入导出和导出不同格式文件,这次是因为有小伙伴问我怎么合并单元格。其实吧很多东西官网https://github.com/SheetJS/js-xlsx讲的比我清楚多了,不过既然问了我也就讲一下吧!!
其他文章传送门:

1.导入数据观察数据格式

1.1.我们先创建一个具有合并单元格的xlsx表格

就以表头数据合并示例吧:


img_1fd54fb65ead9a620dc283c433efbec7.png
示例
1.2.写个简单的数据导入功能(你可以参考前言中的文章编写,我就不放代码了)

导入xlsx参考数据格式:

img_f920dd2439e1583a6847aa86ee828943.png
示例
1.3.查看官网说明

官网https://github.com/SheetJS/js-xlsx#common-spreadsheet-format有做详细说明

img_9f657698cd8b36c0daac9a41e0e36e81.png
官网

官网示例(http://sheetjs.com/demos/modify.html):

img_52edda9da53cabc5c1da835c83f82c51.png
官网示例

根据官网说明我们简单看出合并单元格的数据格式是:

........
data["!merges"] = [{
                s: {//s为开始
                    c: 1,//开始列
                    r: 0//可以看成开始行,实际是取值范围
                },
                e: {//e结束
                    c: 4,//结束列
                    r: 0//结束行
                }
            }];
........

2.动手实验

2.1.写个简单的导出demo

我直接将文章http://www.jianshu.com/p/044c183edf42中的代码改改:
(==>点击查看示例<==)

<!DOCTYPE html>
<html>

<head>
    <meta charset="UTF-8">
    <title></title>
</head>

<body>
    <script src="http://oss.sheetjs.com/js-xlsx/xlsx.full.min.js"></script>
    <!--调用FileSaver saveAs函数可以实现文件下载-->
    <!--<script src="http://sheetjs.com/demos/Blob.js"></script>
    <script src="http://sheetjs.com/demos/FileSaver.js"></script>-->
    <script>
        //如果使用 FileSaver.js 就不要同时使用以下函数
        function saveAs(obj, fileName) {//当然可以自定义简单的下载文件实现方式 
            var tmpa = document.createElement("a");
            tmpa.download = fileName || "下载";
            tmpa.href = URL.createObjectURL(obj); //绑定a标签
            tmpa.click(); //模拟点击实现下载
            setTimeout(function () { //延时释放
                URL.revokeObjectURL(obj); //用URL.revokeObjectURL()来释放这个object URL
            }, 100);
        }
        var jsono = [{ //测试数据
            "id": 1,//A
            "合并的列头1": "数据11",//B
            "合并的列头2": "数据12",//C
            "合并的列头3": "数据13",//D
            "合并的列头4": "数据14",//E
        }, {
            "id": 2,
            "合并的列头1": "数据21",
            "合并的列头2": "数据22",
            "合并的列头3": "数据23",
            "合并的列头4": "数据24",
        }];//....
        const wopts = { bookType: 'xlsx', bookSST: true, type: 'binary' };//这里的数据是用来定义导出的格式类型 
        function downloadExl(data, type) {
            var wb = { SheetNames: ['Sheet1'], Sheets: {}, Props: {} };
            //wb.Sheets['Sheet1'] = XLSX.utils.json_to_sheet(data);//通过json_to_sheet转成单页(Sheet)数据
            data = XLSX.utils.json_to_sheet(data); 
            data["B1"] = { t: "s", v: "asdad" };
            data["!merges"] = [{//合并第一行数据[B1,C1,D1,E1]
                s: {//s为开始
                    c: 1,//开始列
                    r: 0//开始取值范围
                },
                e: {//e结束
                    c: 4,//结束列
                    r: 0//结束范围
                }
            }];
            wb.Sheets['Sheet1'] = data;
            saveAs(new Blob([s2ab(XLSX.write(wb, wopts))], { type: "application/octet-stream"}), "这里是下载的文件名" + '.' + (wopts.bookType == "biff2" ? "xls" : wopts.bookType));
        }
        function s2ab(s) {
            if (typeof ArrayBuffer !== 'undefined') {
                var buf = new ArrayBuffer(s.length);
                var view = new Uint8Array(buf);
                for (var i = 0; i != s.length; ++i) view[i] = s.charCodeAt(i) & 0xFF;
                return buf;
            } else {
                var buf = new Array(s.length);
                for (var i = 0; i != s.length; ++i) buf[i] = s.charCodeAt(i) & 0xFF;
                return buf;
            }
        }
    </script>
    <button onclick="downloadExl(jsono)">导出</button>
</body>
</html>
结合 xlsxUtils 使用示例:

<!DOCTYPE html>
<html>

<head>
    <meta charset="UTF-8">
    <title></title>
</head>

<body>
    <!--以下js地址自行修改-->
    <script src="./xlsx.full.min.js"></script>
    <script src="xlsx.utils.js"></script>
    <!--调用FileSaver saveAs函数可以实现文件下载-->
    <!--<script src="http://sheetjs.com/demos/Blob.js"></script>
    <script src="http://sheetjs.com/demos/FileSaver.js"></script>-->
    <script>
        //如果使用 FileSaver.js 就不要同时使用以下函数
        function saveAs(obj, fileName) {//当然可以自定义简单的下载文件实现方式 
            var tmpa = document.createElement("a");
            tmpa.download = fileName || "下载";
            tmpa.href = URL.createObjectURL(obj); //绑定a标签
            tmpa.click(); //模拟点击实现下载
            setTimeout(function () { //延时释放
                URL.revokeObjectURL(obj); //用URL.revokeObjectURL()来释放这个object URL
            }, 100);
        }
        var jsono = [{ //测试数据
            "id": 1,//A
            "合并的列头1": "数据11",//B
            "合并的列头2": "数据12",//C
            "合并的列头3": "数据13",//D
            "合并的列头4": "数据14",//E
        }, {
            "id": 2,
            "合并的列头1": "数据21",
            "合并的列头2": "数据22",
            "合并的列头3": "数据23",
            "合并的列头4": "数据24",
        }];//....
        const wopts = { bookType: 'xlsx', bookSST: true, type: 'binary' };//这里的数据是用来定义导出的格式类型 
        function downloadExl(data, type) {
            var wb = { SheetNames: ['Sheet1'], Sheets: {}, Props: {} };
            // wb.Sheets['Sheet1'] = XLSX.utils.json_to_sheet(data);//通过json_to_sheet转成单页(Sheet)数据
            data = xlsxUtils.format2Sheet(data);
            data["B1"] = { t: "s", v: "asdad" };
            data["!merges"] = [{//合并第一行数据[B1,C1,D1,E1]
                s: {//s为开始
                    c: 1,//开始列
                    r: 0//开始取值范围
                },
                e: {//e结束
                    c: 4,//结束列
                    r: 0//结束范围
                }
            }]; 
            wb=xlsxUtils.format2WB(data,'Sheet1');
            // data["!ref"]="A1:E7";
            // wb.Sheets['Sheet1'] = data;
            saveAs(new Blob([s2ab(XLSX.write(wb, wopts))], { type: "application/octet-stream" }), "这里是下载的文件名" + '.' + (wopts.bookType == "biff2" ? "xls" : wopts.bookType));
        }
        function s2ab(s) {
            if (typeof ArrayBuffer !== 'undefined') {
                var buf = new ArrayBuffer(s.length);
                var view = new Uint8Array(buf);
                for (var i = 0; i != s.length; ++i) view[i] = s.charCodeAt(i) & 0xFF;
                return buf;
            } else {
                var buf = new Array(s.length);
                for (var i = 0; i != s.length; ++i) buf[i] = s.charCodeAt(i) & 0xFF;
                return buf;
            }
        }
    </script>
    <button onclick="downloadExl(jsono)">导出</button>
</body>

</html>
2.2.最终效果
img_503dbb47c410a86effb49ac0d8d72e73.gif
gif
img_0bef5532db71cce8339e008729f08a60.png
效果

合并功能就说到这里吧!至于导入的读取我相信小伙伴们也应该知道怎么处理了吧!
当然这里顺便解决下其他小伙伴问的关于xlsx数据量大的问题吧!

  • 对于导入:本身前端处理xlsx效率大多情况下没有后端效率高。原因是大多数客户端机器性能并不是很高,有可能数据量大就浏览器未响应了。所以我是在导入时会对文件大小和数据量进行判断。当然我建议你可以根据实际业务判断使用切换为前端处理还是后端处理文件。(其实我觉的用python处理也挺简单的)
  • 对于导出:况且大xlsx文件有点机子可能连打都打不开更别说编辑了。我之所以选用js-xlsx原因第一是降低服务器压力,第二就是为了防止导出数据量太大一些客户端电脑太差连文件都打不开编辑,所以限制文件导出数据量,分成多个xlsx导出。况且json处理与传输也比较简单。
总之最好是根据实际业务选择好的解决方案

版权声明:本文内容由阿里云实名注册用户自发贡献,版权归原作者所有,阿里云开发者社区不拥有其著作权,亦不承担相应法律责任。具体规则请查看《阿里云开发者社区用户服务协议》和《阿里云开发者社区知识产权保护指引》。如果您发现本社区中有涉嫌抄袭的内容,填写侵权投诉表单进行举报,一经查实,本社区将立刻删除涉嫌侵权内容。

相关文章
阿里云ECS部署Grafana接入zabbix
Grafana接入zabbix部署 阿里云ECS部署Grafana接入zabbix Grafana 是 Graphite 和 InfluxDB 仪表盘和图形编辑器。Grafana 是开源的,功能齐全的度量仪表盘和图形编辑器,支持 Graphite,InfluxDB 和 OpenTSDB。
4363 0
WebGIS中前端JS生成等值面方法探讨
文章版权由作者李晓晖和博客园共有,若转载请于明显处标明出处:http://www.cnblogs.com/naaoveGIS/ 1.背景         在之前的博文《WebGIS中等值面展示的相关方案简析》中我提到了两种生成等值面的方法:        a.使用GP服务发布等值面生成服务,前端调用该服务生成等值面图片然后叠加至地图上。
1615 0
ExtJS前端实现过程
从无到有的实现过程
1595 0
网站常见问题1分钟定位 - 如何使用阿里云ARMS诊断Java应用卡顿问题
    不要慌,上面只是一张贴图。     为什么“慢”那么难查   网站卡顿、页面加载过慢是互联网应用最常见的问题之一。排查、解决这类问题通常会花费开发运维人员大量的时间,通常是因为以下三个原因: 应用链路太长,无从下手。
1962 0
23.Eclipse下Ndk开发(OpenSL ES播放音频wav)
OpenSL ES(Open Sound Library for Embedded Systems) 网上解释OpenSL ES是跨平台、针对嵌入式系统精心优化的硬件音频加速API。
766 0
如何用纯 CSS 创作牛奶文字变换效果
效果预览 在线演示 按下右侧的“点击预览”按钮可以在当前页面预览,点击链接可以全屏预览。 https://codepen.io/comehope/pen/MGNWOm 可交互视频教程 此视频是可以交互的,你可以随时暂停视频,编辑视频中的代码。
630 0
前端经典面试题解密:JS的new关键字都干了什么?
new关键字在实例化获取对象时都做了什么?是一道经常出现在前端面试时的问题。如果只是简单的了解new关键字是实例化构造函数获取对象,是万万不能够的。更深入的层级发生了什么呢?同时面试官想从这道题里面考察什么呢?下面胡哥为各位小伙伴一一来解密。
0 0
前端项目实战86-ant design table合并单元格
前端项目实战86-ant design table合并单元格
0 0
【前端】JSX 和 HTML 的区别
【前端】JSX 和 HTML 的区别
0 0
+关注
关爱单身狗
简书:https://www.jianshu.com/u/f19e29243ff6
文章
问答
文章排行榜
最热
最新
相关电子书
更多
低代码开发师(初级)实战教程
立即下载
阿里巴巴DevOps 最佳实践手册
立即下载
冬季实战营第三期:MySQL数据库进阶实战
立即下载