SpringBoot+layui实现Excel导入操作

news2024/11/17 20:42:39

excel导入步骤

    • 第三方插件
      • 引入插件
    • 效果图 (方法1)
    • 代码实现(方法1)
      • Html代码( 公共)
        • 下载导入模板
      • js实现 (方法1)
        • 上传文件实现
      • 效果图(方法2)
      • 代码实现(方法2)这里主要改的是js部分
      • 后端代码(公共部分)
        • controller
        • Service
        • serviceImpl
        • mapper
        • mapper.xml

第三方插件

layui_Excel

引入插件

 layui.use(['form', 'table', 'notify', 'dtree', 'dropdown', 'excel', 'element', 'laytpl'], function () {
        var $ = layui.jquery,
            form = layui.form,
            table = layui.table;
        var notify = layui.notify;
        var util = layui.util;
        var laydate = layui.laydate;
        var excel = layui.excel;
        var laytpl = layui.laytpl;
        var element = layui.element;
        
})

效果图 (方法1)

在这里插入图片描述

代码实现(方法1)

Html代码( 公共)

在这里插入图片描述

  <button class="layui-btn data-light-btn layui-btn-sm " lay-event="data-export-btn"><i
                                class="fa fa-download"></i> 导入
                        </button>

在这里插入图片描述


<div id="file" style="display: none;padding:20px;">
    <div class="layui-form-item">
        <div class="layui-form-block">
            <input type="file" class="layui-btn data-white-btn" id="LAY-excel-import-excel"
                   multiple="multiple">
            <!--提示-->
            <p>
            <div class="layui-form-mid layui-text-em">
                <span style="color: #FF5722;">*</span>仅允许导入“xls”或“xlsx”格式文件!
                <!--下载导入模板-->
                <button class="layui-btn layui-btn-xs data-gray-btn" lay-on="LAY-excel-import-download">
                    <i class="fa fa-file-excel-o"></i>下载模板
                </button>
            </div>
            </p>
        </div>
    </div>
</div>
下载导入模板

在这里插入图片描述

  util.on("lay-on", {
           
            "LAY-excel-import-download": function () {//导入Excel
                //下载导入模板
                window.location.href = "../dist/erpInfo/用户数据导入模板.xlsx";
                //设置下载名称
                var fileName = "用户数据导入模板.xlsx";
            }

        })

js实现 (方法1)

上传文件实现

在这里插入图片描述

 table.on('toolbar(userInfoTableFilter)', function (obj) {
            var id = obj.config.id;
            var checkStatus = table.checkStatus(id);
            var dataArr = checkStatus.data;//用于删除
            var data = checkStatus.data[0]; //获取选中行数据
            var othis = lay(this);
            switch (obj.event) {
                 case 'data-export-btn':
                    layer.open({
                        type: 1,
                        title: '导入数据',
                        shadeClose: true,
                        content: $('#file'),
                        skin: 'class-layer-sea',
                        btn: ['<i class="fa fa-check"></i> 导入', '<i class="fa fa-times"></i> 取消'],
                        yes: function (index) {
                            //判断当前是否有文件
                            if (document.getElementById('LAY-excel-import-excel').files.length === 0) {
                                notify.info('请选择需要导入的文件文件', "vcenter", "shadow", false, 1000);
                                return false;
                            }
                            //判断文件格式是否xls”或“xlsx
                            if (document.getElementById('LAY-excel-import-excel').files[0].name.split('.')[1] !== 'xls' && document.getElementById('LAY-excel-import-excel').files[0].name.split('.')[1] !== 'xlsx') {
                                notify.warning('仅允许导入“xls”或“xlsx”格式文件!', "vcenter", "shadow", false, 1000);
                                return false;
                            }
                            //获取当前文件
                            var files = document.getElementById('LAY-excel-import-excel').files;
                            //读取文件
                            excel.importExcel(files, {
                                // 可以在读取数据同时梳理数据
                            }, function (data, book) {
                                // 也可以全部读取出来再进行数据梳理
                                data = excel.filterImportData(data, {
                                    'nickname': 'A'
                                    , 'account': 'B'
                                    , 'email': 'C'
                                    , 'deptId': 'D'
                                    , 'postId': 'E'
                                    , 'gender': 'F'
                                    , 'status': 'G'
                                })
                                let getdata;//选择的表数据
                                $.each(data[0], function (index, value) {
                                    if (getdata == undefined) {
                                        getdata = data[0][index];//从第二行开始获取数据
                                    }
                                })
                                //tab选择,采集上传数据
                                element.on('tab(test)', function (d) {
                                    getdata = data[0][this.getAttribute('lay-id')];//第一个表格数据
                                });
                                // 打印测试数据:console.log(JSON.stringify(getdata.slice(1)))
                                // 判断是否为空数据
                                if (JSON.stringify(getdata.slice(1))==='[]'){
                                    notify.warning('不能导入空数据!', "vcenter", "shadow", false, 1000);
                                    return false;
                                }
                                notify.loading('正在导入数据,请稍后...', "vcenter", "shadow", false)
                                setTimeout(function () {
                                    notify.destroyAll();//关闭加载框
                                    $.ajax({
                                        url: '/user/importUserInfo',
                                        type: 'post',
                                        data: {
                                            name: JSON.stringify(getdata.slice(1))
                                        },
                                        success: function (res) {
                                            if (res.code === 0) {
                                                notify.success(res.msg, "vcenter", "shadow", false, 1000);
                                            }else if (res.code === 1) {
                                                notify.warning(res.msg, "vcenter", "shadow", false, 1000);
                                            }else {
                                                notify.error(res.msg, "vcenter", "shadow", false, 1000);
                                            }
                                        }
                                    }).done(function () {
                                        setTimeout(function () {
                                            notify.destroyAll();
                                            layer.closeAll();
                                            parent.location.reload();//重载页面
                                        }, 500);
                                    });
                                },2000)
                            })
                        },
                        btn2: function () {
                            $('#file').find('input').val('');
                            layer.closeAll();
                            window.location.reload();
                            return false;
                        }

                    });
                    break;
            }
            ;
        });

效果图(方法2)

在这里插入图片描述

代码实现(方法2)这里主要改的是js部分

 table.on('toolbar(userInfoTableFilter)', function (obj) {
            var id = obj.config.id;
            var checkStatus = table.checkStatus(id);
            var dataArr = checkStatus.data;//用于删除
            var data = checkStatus.data[0]; //获取选中行数据
            var othis = lay(this);
            switch (obj.event) {
            case 'data-export-btn':
                    layer.open({
                        type: 1,
                        title: '导入数据',
                        shadeClose: true,
                        content: $('#file'),
                        skin: 'class-layer-sea',
                        btn: ['<i class="fa fa-times"></i> 取消'],
                        success: function () {
                            $(function () {
                                // 监听上传文件的事件
                                $('#LAY-excel-import-excel').change(function (e) {
                                    // 注意:这里直接引用 e.target.files 会导致 FileList 对象在读取之前变化,导致无法弹出文件
                                    var files = Object.values(e.target.files)
                                    uploadExcel(files)
                                    // 变更完清空,否则选择同一个文件不触发此事件
                                    e.target.value = ''
                                })
                                // 文件拖拽
                                document.body.ondragover = function (e) {
                                    e.preventDefault()
                                }
                                document.body.ondrop = function (e) {
                                    e.preventDefault()
                                    var files = e.dataTransfer.files
                                    uploadExcel(files)
                                }

                            })
                        },
                        btn2: function () {
                            $('#file').find('input').val('');
                            layer.closeAll();
                            window.location.reload();
                            return false;
                        }

                    });
                    break;
            }
            ;
        });


        /**
         * 上传excel的处理函数,传入文件对象数组
         * @param  {FileList} files [description]
         * @return {[type]}       [description]
         * 方法2:
         */

        function uploadExcel(files) {
            try {
                excel.importExcel(files, {
                    // 可以在读取数据同时梳理数据
                }, function (data, book) {
                    // 也可以全部读取出来再进行数据梳理
                    data = excel.filterImportData(data, {
                        'nickname': 'A'
                        , 'account': 'B'
                        , 'email': 'C'
                        , 'deptId': 'D'
                        , 'postId': 'E'
                        , 'gender': 'F'
                        , 'status': 'G'
                    })
                    let getdata;
                    $.each(data[0], function (index, value) {
                        if (getdata == undefined) {
                            getdata = data[0][index];//从第二行开始获取数据
                        }
                    })
                    //tab选择,采集上传数据
                    element.on('tab(test)', function (d) {
                        getdata = data[0][this.getAttribute('lay-id')];//第一个表格数据

                    });
                    // 如果不需要展示直接上传,可以再次 $.ajax() 将JSON数据通过 JSON.stringify() 处理后传递到后端即可
                    layer.open({
                        title: '文件转换结果检阅',
                        closeBtn: false,
                        skin: 'class-layer-yellow',
                        area: ['630px', '300px'],
                        tipsMore: true,
                        content: laytpl($('#LAY-excel-export-ans').html()).render({data: data, files: files}),
                        success: function () {
                            element.render('tab');
                            layui.code({});
                            // 处理合并
                            for (var file_index in book) {
                                if (!book.hasOwnProperty(file_index)) {
                                    continue;
                                }
                                // 遍历每个Sheet
                                for (var sheet_name in book[file_index].Sheets) {
                                    if (!book[file_index].Sheets.hasOwnProperty(sheet_name)) {
                                        continue;
                                    }
                                    var sheetObj = book[file_index].Sheets[sheet_name];
                                    // 仅在有合并参数时进行操作
                                    if (!sheetObj['!merges']) {
                                        continue;
                                    }
                                    // 遍历每个Sheet中每个 !merges
                                    for (var merge_index = 0; merge_index < sheetObj['!merges'].length; merge_index++) {
                                        var mergeObj = sheetObj['!merges'][merge_index];
                                        // 每个合并参数的 s.c 表示左上角单元格的列,s.r 表示左上角单元格的行,e.c 表示右下角单元格的列,e.r 表示右下角单元格的行,计算时注意 + 1
                                        $('#table-export-' + file_index + '-' + sheet_name + '-' + mergeObj.s.r + '-' + mergeObj.s.c)
                                            .prop('rowspan', mergeObj.e.r - mergeObj.s.r + 1)
                                            .prop('colspan', mergeObj.e.c - mergeObj.s.c + 1);
                                        for (var r = mergeObj.s.r; r <= mergeObj.e.r; r++) {
                                            for (var c = mergeObj.s.c; c <= mergeObj.e.c; c++) {
                                                // 排除左上角
                                                if (r === mergeObj.s.r && c === mergeObj.s.c) {
                                                    continue;
                                                }
                                                $('#table-export-' + file_index + '-' + sheet_name + '-' + r + '-' + c).remove();
                                            }
                                        }
                                    }
                                }
                            }
                        },
                        btn: ['立即导入', '放弃导入'],
                        yes: function (index, layero) {
                            if (JSON.stringify(getdata.slice(1))==='[]'){
                                    notify.warning('不能导入空数据!', "vcenter", "shadow", false, 1000);
                                    return false;
                                }
                                notify.loading('正在导入数据,请稍后...', "vcenter", "shadow", false)
                                setTimeout(function () {
                                    notify.destroyAll();//关闭加载框
                                    $.ajax({
                                        url: '/user/importUserInfo',
                                        type: 'post',
                                        data: {
                                            name: JSON.stringify(getdata.slice(1))
                                        },
                                        success: function (res) {
                                            if (res.code === 0) {
                                                notify.success(res.msg, "vcenter", "shadow", false, 1000);
                                            }else if (res.code === 1) {
                                                notify.warning(res.msg, "vcenter", "shadow", false, 1000);
                                            }else {
                                                notify.error(res.msg, "vcenter", "shadow", false, 1000);
                                            }
                                        }
                                    }).done(function () {
                                        setTimeout(function () {
                                            notify.destroyAll();
                                            layer.closeAll();
                                            parent.location.reload();//重载页面
                                        }, 500);
                                    });
                                },2000)
                            return false;
                        },
                        btn2: function (index, layero) {
                            // 点击取消按钮的回调函数
                            layer.msg("取消上传成功!");
                            layer.close(index); // 关闭弹窗
                        }
                    });
                })
            } catch (e) {
                layer.alert(e.message)
            }
        }

后端代码(公共部分)

controller

这里处理JSON的依赖

 <!--阿里巴巴fastjson依赖-->
        <dependency>
            <groupId>com.alibaba</groupId>
            <artifactId>fastjson</artifactId>
            <version>1.2.83_noneautotype</version>
        </dependency>
/*
    * 数据导入操作
    * @importUserInfo
    * */
    @PostMapping("/importUserInfo")
    public  Map<String, Object> importUserInfo(@RequestParam("name") String data) {
        // 创建一个HashMap对象,用于存储返回的结果
        Map<String, Object> resultMap = new HashMap<>();
        System.out.println("导入用户信息");
        try {
            // 解析前端传递的JSON数据
            List<UserEntity> userEntities = JSON.parseArray(data, UserEntity.class);
            // 判断导入的文件中手机号是否在数据库中存在
            for (UserEntity user : userEntities) {
                // 判断导入的文件中手机号是否在数据库中存在
                UserEntity existingUser = userInfoService.selectByAccount(user.getAccount());
                if (existingUser != null) {
                    resultMap.put("code", 1);
                    resultMap.put("msg", "该"+ user.getAccount()+"账户已存在!");
                    return resultMap;
                }
                // 设置创建时间
                user.setCreateTime(new Date());
                // 设置默认头像
                user.setAvatar(user.getGender() == 1 ? "images/avatar/69292338265a201591b9412c9feb6c192788f21a.jpeg" : "images/avatar/89219682b88c376bddc145ca9a9f6d66fddbbd28.jpeg");
                //设置密码为手机号后六位
                user.setPassword(MD5Util.encrypt(user.getAccount().substring(5)));
                //System.out.println(user.getAccount().substring(5));
                // 设置角色ID
                user.setRoleId(3);
            }
            // 调用Service层方法将数据插入数据库
            userInfoService.insertUserInfoExcel(userEntities);
            resultMap.put("code", 0);
            resultMap.put("msg", "数据导入成功!");
        } catch (Exception e) {
            resultMap.put("msg", "文件不合法,导入失败!" + e.getMessage());
        }
        return resultMap;
    }
Service
   /*
    * excel导入
    * */
    void insertUserInfoExcel(List<UserEntity> userEntities);
serviceImpl

    /*
    * excel导入
    * */
    @Override
    public void insertUserInfoExcel(List<UserEntity> userEntities) {
        userMapper.insertUserInfoExcel(userEntities);
    }
mapper
  /*
    * excel导入
    * @insertUserInfoExcel
    * */
    void insertUserInfoExcel(List<UserEntity> userEntities);
mapper.xml
  <!--excel导入-->
    <insert id="insertUserInfoExcel" parameterType="java.util.List">
        insert into sys_user (nickname,account,email,dept_id,post_id,gender,status,password,roleId,avatar,createTime)
        values
        <foreach item="item" collection="list" separator=",">
            (#{item.nickname},#{item.account},#{item.email},#{item.deptId},#{item.postId},#{item.gender},#{item.status},#{item.password},#{item.roleId},#{item.avatar},#{item.createTime})
        </foreach>
    </insert>

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1721384.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

一碗米线火了24年,蒙自源六一再献新作

当一碗热气腾腾的米线在餐桌上飘香四溢&#xff0c;你是否会想起那个陪伴了无数食客24年的名字——蒙自源&#xff1f;在这个充满欢笑与童真的六一儿童节&#xff0c;蒙自源米线品牌再度发力&#xff0c;用全新的儿童餐系列为孩子们带来了一份特别的节日礼物。 蒙自源&#xf…

性价比为王,物流商怎么选择高效的国际物流管理平台

在全球化贸易日益繁荣的今天&#xff0c;国际物流行业作为链接国内商家和海外市场的重要桥梁&#xff0c;发挥着极其重要的作用。 然而&#xff0c;随着国际物流市场竞争的加剧&#xff0c;对物流商来说&#xff0c;也面临着成本管控和效率提升的双重挑战。今天我们会重点探讨…

AI之下 360让PC商业生态大象起舞

时隔7年&#xff0c;淘宝PC版在前不久迎来重磅升级&#xff0c;在产品体验、商品供给、内容供给等方面做了全面优化&#xff0c;以全面提升PC端的用户体验&#xff1b;当大家都以为移动互联网时代下APP将成为主流时&#xff0c;PC端却又成为了香饽饽。其实PC端被重视&#xff0…

【Qt】【模型视图架构】代理模型示例

文章目录 1. 基本排序/过滤模型Basic Sort/Filter Model Example2. 自定义排序/过滤模型Custom Sort/Filter Model ExampleFilterLineEdit类定义及实现MySortFilterProxyModel类定义及实现 1. 基本排序/过滤模型Basic Sort/Filter Model Example 官方提供的基本排序/过滤模型示…

LeetCode42:接雨水

题目描述 给定 n 个非负整数表示每个宽度为 1 的柱子的高度图&#xff0c;计算按此排列的柱子&#xff0c;下雨之后能接多少雨水。 代码 单调栈 class Solution { public:int trap(vector<int>& height) {stack<int> stk;int result 0;stk.push(0);for (int …

Android设备获取OAID调研和实现

什么是OAID、AAID、VAID OAID OAID是"Android ID"&#xff08;安卓ID&#xff09;的一种替代方案&#xff0c;其全称为"Open Anonymous Identifier"&#xff08;开放匿名标识符&#xff09;。 因传统的移动终端设备标识如国际移动设备识别码&#xff08;…

【Python内功心法】:深挖内置函数,释放语言潜能

文章目录 &#x1f680;一、常见内置函数&#x1f308;二、高级内置函数⭐1. enumerate函数&#x1f44a;2. eval函数❤️3. exec函数&#x1f4a5;4. eval与exec 中 globals与locals如何用☔4-1 globals 参数&#x1f3ac;4-2 locals 参数 ❤️5. filter函数&#x1f44a;6. z…

发电机组故障的原因、解决方案及解决措施

发电机组故障的原因、解决方案及解决措施可以总结如下&#xff1a; 一、故障原因 供电中断 原因&#xff1a;电网故障、线路短路或电力负荷过重等。 燃油问题 原因&#xff1a;燃油供应系统问题&#xff0c;如燃油管路堵塞、燃油质量不佳等。 轴承过热 原因&#xff1a;轴承过…

小学生四则运算练习器,用户可以选择进行加减乘除任意一项,也可以选择退出,然后每次计算后会提示正确与否,最后计算总分然后并给出评语。

⑴ 用户可以从菜单中选择某种运算进行练习。具体包括&#xff1a;加法&#xff0c;减法&#xff0c;乘法&#xff0c;除法&#xff1b;也可以退出程序. ⑵ 用户可以指定每次练习的题目数量、设置练习的总分&#xff1b; ⑶ 每小题练习后给出结果正确与否的提示&#xff1b;一…

【机器学习】智能选择的艺术:决策树在机器学习中的深度剖析

在机器学习的分类和回归问题中&#xff0c;决策树是一种广泛使用的算法。决策树模型因其直观性、易于理解和实现&#xff0c;以及处理分类和数值特征的能力而备受欢迎。本文将解释决策树算法的概念、原理、应用、优化方法以及未来的发展方向。 &#x1f680;时空传送门 &#x…

基于单片机的微型嵌入式温度测量仪的设计与实现分析

摘要 &#xff1a; 作为信息技术中重要的技术手段之一嵌入式单片机系统已经被应用到越来越多不同的行业领域中。如&#xff0c;各种手持监测设备、智能家电设备等。当前展开对单片机的微型嵌入式温度测量仪的设计和实现研究&#xff0c;从微型嵌入式单片机相关理论入手&#xf…

Java Apache Jaccard文本相似度匹配初体验

文章目录 前言一、文本相似度算法的选择二、常见的文本相似度算法介绍三、使用示例1、引入jar包2、方法示例3、Jaccard源码剖析4、Jaccard源码解释 写在最后 前言 产品今天提了个需求&#xff0c;大概是这样的&#xff0c;来&#xff0c;请看大屏幕。。。额。。。搞错了&#…

IDEA插件开发:自动生成setter

背景 在给Java局部变量的实体赋值时&#xff0c;往往有很多setter&#xff0c;一个一个写很麻烦&#xff0c;也会漏掉&#xff0c;因此开发一款插件&#xff0c;可以自动生成局部变量实体的所有setter。 插件效果如下&#xff1a; 可以在plugin marketplace 搜索&#xff1…

Nginx 1.26.0 爆 HTTP/3 QUIC 漏洞,建议升级更新到 1.27.0

据悉&#xff0c;Nginx 1.25.0-1.26.0 主线版本中涉及四个与 NGINX HTTP/3 QUIC 模块相关的中级数据面 CVE 漏洞&#xff0c;其中三个为 DoS 攻击类型风险&#xff0c;一个为随机信息泄漏风险&#xff0c;影响皆为允许未经身份认证的用户通过构造请求实施攻击。目前已经紧急发布…

【ARM-Linux篇】u-boot编译

一、u-boot简介 uboot是一种通用的引导加载程序&#xff0c;它可以用于多种嵌入式系统&#xff0c;支持多种操作系统&#xff0c;如Linux, Android,NetBSD等。uboot的主要作用是将操作系统内核从存储设备&#xff08;如Flash, SD卡等&#xff09;加载到内存中&#xff0c;并执…

Thread的stop和interrupt的区别

Thread.stop Thread.stop()方法已被废弃。 因为本质上它是不安全的&#xff0c;使用该方法可能会导致数据、资源不一致的问题&#xff0c; public class ThreadDemo {static class MyThread extends Thread {Overridepublic void run() {while (true) {try {Thread.sleep(10…

C#WPF数字大屏项目实战03--数据内容区域

1、内容区域划分 第一行标题&#xff0c;放了几个文本框 第二行数据&#xff0c;划分成3列布局 2、第1列布局使用UniformGrid控件 最外面放UniformGrid&#xff0c;然后里面放3个GroupBox控件&#xff0c;这3个groupbox都是垂直排列 3、GroupBox控件模板 页面上的3个Group…

基于SSM的“健身俱乐部网站”的设计与实现(源码+数据库+文档)

基于SSM的“健身俱乐部网站”的设计与实现&#xff08;源码数据库文档) 开发语言&#xff1a;Java 数据库&#xff1a;MySQL 技术&#xff1a;SSM 工具&#xff1a;IDEA/Ecilpse、Navicat、Maven 系统展示 系统功能结构图 用户注册界面图 系统登录界面 添加管理员账户界面…

微服务架构-微服务实施

目录 一、概述 二、微服务拆分 2.1 概述 2.2 拆分原则 2.3 拆分方法 2.3.1 以数据为维度进行拆分 2.3.2 按照使用场景拆分 2.3.3 重要和非重要的拆分 2.3.4 变和不变的拆分 三、微服务通信 3.1 概述 3.2 微服务通信方式选择 3.3 微服务编排 3.4 API接口设计 3.5 …

CANDela studio新建和编辑服务

服务定义和编辑只能够在CDDT里面进行&#xff0c;思路分为三步&#xff1a; 1、Protocol Services里面添加服务&#xff0c;定义服务的格式、请求和正负响应。 2、根据服务的功能归类到Diagnostic Class Tenplates 3、Variant里面的Supported Diagnostic Classes勾选 然后我…