前后端的导入、导出、模板下载等写法

导入,导出、模板下载等的前后端写法

文章目录

  • 导入,导出、模板下载等的前后端写法
    • 一、导入实现
      • 1.1 后端的导入
      • 1.2 前端的导入
    • 二、基础的模板下载
      • 2.1 后端的模板下载-若依基础版本
      • 2.2 前端的模板下载
      • 2.3 后端的模板下载 - 基于资源文件读取
      • 2.4 excel制作下载模板
    • 三、结语

闲来无事,自己写了一个小demo,向着全栈再迈出一小步,而且这些demo以后工作也可以用到。这个目前都是基于若依写的。

一、导入实现

1.1 后端的导入

后端我就只列举核心的方法了,简单的sql就不加上来了。

BookController

    // 导入方法
    @PostMapping("/importData")
    @ResponseBody
    public AjaxResult importData(MultipartFile file, boolean updateSupport) throws Exception
    {
        ExcelUtil<Book> util = new ExcelUtil<Book>(Book.class);
        List<Book> bookList = util.importExcel(file.getInputStream());
        AjaxResult result = bookService.importBookList(bookList);
        return result;
    }

BookServiceImpl

下面这一段才是我写的导入的核心,以后改改就可以直接拿来用的

@Override
    @Transactional(rollbackFor = Exception.class)
    public AjaxResult importBookList(List<Book> bookList) {
        //========= 可有可无,如果哪一列必填项为空,就中断程序,并抛出异常 ======================
        if (CollectionUtils.isEmpty(bookList)) {
            throw new ServiceException("导入书籍数据不能为空!");
        }
        List<Integer> errorRows = new ArrayList<Integer>();
        for (int i = 0; i < bookList.size(); i++) {
            if (StringUtils.isEmpty(bookList.get(i).getBookName())) {
                int rowNumber = i + 1;
                errorRows.add(rowNumber);
            }
        }
        if (errorRows.size() > 0) {
            StringBuilder errorMessage = new StringBuilder();
            errorMessage.append("导入书籍数据不能为空,以下行的书籍名称为空:");
            for (int row : errorRows) {
                errorMessage.append("第").append(row).append("行、");
            }
            errorMessage.deleteCharAt(errorMessage.length() - 1);
            throw new ServiceException(errorMessage.toString());
        }
        //============= 可有可无,如果哪一列必填项为空,就中断程序,并抛出异常 ==================
        //============= 逐行校验导入 ===================================================
        int successNum = 0;
        int failureNum = 0;
        //用于判断第几行出现了错误
        int row = 0;
        StringBuilder successMsg = new StringBuilder();
        StringBuilder failureMsg = new StringBuilder();
        for (Book book : bookList) {
            try {
                //根据书籍名称查询是否已存在
                if (ObjectUtils.isNotEmpty(book.getBookName())) {
                    Book bookSize = bookMapper.selectBookByName(book.getBookName());
                    if (ObjectUtils.isEmpty(bookSize)) {
                        book.setCreateBy(SecurityUtils.getUsername());
                        book.setCreateTime(new Date());
                        bookMapper.insertBook(book);
                        successNum++;
                        successMsg.append("<br/>" + "书籍信息" + book.getBookName() + " 导入成功");
                    } else {
                        book.setUpdateTime(new Date());
                        book.setUpdateBy(SecurityUtils.getUsername());
                        bookMapper.updateBookName(book);
                        successNum++;
                        successMsg.append("<br/>" + "书籍信息" + book.getBookName() + " 更新成功");
                    }
                    row ++;
                }
            } catch (Exception e) {
                failureNum++;
                String msg = "<br/>" + failureNum + "书籍名称为:" + book.getBookName() + " 的数据导入失败:"+"其行号为"+ row ;
                failureMsg.append(msg + e.getMessage());
                log.error(e.getMessage(), e);
            }
        }
        if (failureNum > 0) {
            failureMsg.insert(0, "很抱歉,导入失败!共 " + failureNum + " 条数据格式不正确,错误如下:");
            throw new ServiceException(failureMsg.toString());
        } else {
            successMsg.insert(0, "恭喜您,数据已全部导入成功!共 " + successNum + " 条,数据如下:");
        }
        return AjaxResult.success(successMsg);
    }

1.2 前端的导入

这里前端我就不分三大块了,相信都看得懂应该加在哪

      <el-col :span="1.5">
        <el-button
          type="info"
          plain
          icon="el-icon-upload2"
          size="mini"
          @click="handleImport"

        >导入</el-button>

 <!--书籍导入对话框-->
    <el-dialog :title="upload.title" :visible.sync="upload.open" width="400px" append-to-body>
      <el-upload
        ref="upload"
        :limit="1"
        accept=".xlsx, .xls"
        :headers="upload.headers"
        :action="upload.url + '?updateSupport=' + upload.updateSupport"
        :disabled="upload.isUploading"
        :on-progress="handleFileUploadProgress"
        :on-success="handleFileSuccess"
        :auto-upload="false"
        drag
      >
        <i class="el-icon-upload"></i>
        <div class="el-upload__text">将文件拖到此处,或<em>点击上传</em></div>
        <div class="el-upload__tip text-center" slot="tip">
          <div class="el-upload__tip" slot="tip">
            <el-checkbox v-model="upload.updateSupport" /> 是否更新已经存在的用户数据
          </div>
          <span>仅允许导入xls、xlsx格式文件。</span>
          <el-link type="primary" :underline="false" style="font-size:12px;vertical-align: baseline;" @click="exportTemplate">下载模板</el-link>
        </div>
      </el-upload>
      <div slot="footer" class="dialog-footer">
        <el-button type="primary" @click="submitFileForm">确 定</el-button>
        <el-button @click="upload.open = false">取 消</el-button>
      </div>
    </el-dialog>

  =============================
     写在data里面
 // 书籍导入参数
      upload: {
        // 是否显示弹出层(用户导入)
        open: false,
        // 弹出层标题(用户导入)
        title: "",
        // 是否禁用上传
        isUploading: false,
        // 是否更新已经存在的用户数据
        updateSupport: 0,
        // 设置上传的请求头部
        headers: { Authorization: "Bearer " + getToken() },
        // 上传的地址
        // url: process.env.VUE_APP_BASE_API + "/system/user/importData"
        url:  process.env.NODE_ENV === "production"
          ? getBaseUrl() + "/system/book/importData"
          : process.env.VUE_APP_BASE_API + "/system/book/importData"

      },
=================================
    写在method里面
    /** 导入按钮操作 */
    handleImport() {
      this.upload.title = "用户导入";
      this.upload.open = true;
    },
    // 文件上传中处理
    handleFileUploadProgress(event, file, fileList) {
      this.upload.isUploading = true;
      // this.$set(this.upload,'open',true)
    },
 // 文件上传成功处理
    handleFileSuccess(response, file, fileList) {
      debugger;
      this.upload.open = false;
      this.upload.isUploading = false;
      // this.$set(this.upload,'open',false)
      this.$refs.upload.clearFiles();
      let previewText = response.data.length > 100 ? response.data.substring(0, 100)+ "..." : response.data;
      this.$alert(previewText, "导入结果", { dangerouslyUseHTMLString: true });
      this.getList();
    },
// 提交上传文件
    submitFileForm() {
      this.$refs.upload.submit();
    },
  }     

二、基础的模板下载

这里的模板下载写法是和数据导出的写法是一样的,但是我想再加一个读取资源文件中已经定义的模板进行下载

2.1 后端的模板下载-若依基础版本

这里用的是若依的注解导出空的集合就相当于是一个模板了,但是不好的是模板不能够限定属性字段的格式,不方便导入。后面会介绍另一种 Java基础的读取资源文件的模板下载。

    //下载模板
    @PostMapping("/exportTemplate")
    @ResponseBody
    public void exportTemplate(HttpServletResponse response)
    {
        ExcelUtil<Book> util = new ExcelUtil<Book>(Book.class);
        util.importTemplateExcel(response,"书籍数据");
    }

Book.java

@Data
public class Book extends BaseEntity
{
    private static final long serialVersionUID = 1L;

    /** 主键 */
    private Long id;

    /** 书名 */
    @Excel(name = "书名")
    private String bookName;

    /** 图书类型(1:名著,2:历史,3:社科) */
    @Excel(name = "图书类型", readConverterExp = "1=:名著,2:历史,3:社科")
    private String bookType;

    /** 作者 */
    @Excel(name = "作者")
    private String author;

    /** 编辑 */
    @Excel(name = "编辑")
    private String edithor;

    /** 出版日期 */
    @JsonFormat(pattern = "yyyy-MM-dd")
    @Excel(name = "出版日期", width = 30, dateFormat = "yyyy-MM-dd")
    private Date publishDate;

    /** 出版社 */
    @Excel(name = "出版社")
    private String publisher;

    /** 售价 */
    @Excel(name = "售价")
    private BigDecimal price;

2.2 前端的模板下载

这里就用的是若依的基础下载方式

      <el-button
          type="warning"
          plain
          icon="el-icon-download"
          size="mini"
          @click="exportTemplate"
        >模板下载</el-button>
        
   下面这个其实就是若依整了一个全局的下载方法     
    exportTemplate() {
      this.download('system/book/exportTemplate', {
      }, `书籍_${new Date().getTime()}.xlsx`)
      // exportTemplate().then(response => {
      // });
    },

比如在main.js里面全局挂载
import { download } from '@/utils/request'
Vue.prototype.download = download
然后可以具体看看它在request.js里面是怎么封装的

2.3 后端的模板下载 - 基于资源文件读取

基于资源文件读取的方式下载模板可以先定义好excel的格式,比如限制输入字典类型的字段的时候,在excel里面先限制好格式。而且这种方法也比较通用(文件名称前端都是可以设置的) 前端的写法不变

//基于资源文件读取下载模板
    @PostMapping("/exportResourceTemplate")
    public void exportResourceTemplate(HttpServletResponse response)
    {
        ClassPathResource resource = new ClassPathResource("template/book_1720181938018.xlsx");
        InputStream inputStream = null;
        OutputStream outputStream = null;
        try {
            inputStream = resource.getInputStream();
            outputStream = response.getOutputStream();
            response.setContentType("application/vnd.openxmlformats-officedocument.spreadsheetml.sheet");
            String fileName = "书籍模板.xlsx";
            String encodedFileName = UriUtils.encode(fileName, "UTF-8");
            response.setHeader("Content-Disposition", "attachment; filename=\"" + encodedFileName + "\"");
            //这里定义了一个大小为 1024 字节(1 KB)的字节数组 buffer。这个缓冲区用于暂存从 InputStream 中读取的数据。
            byte[] buffer = new byte[1024];
            int bytesRead;
            while ((bytesRead = inputStream.read(buffer)) != -1) {
                outputStream.write(buffer, 0, bytesRead);
            }
        } catch (IOException e) {
            e.printStackTrace();
            // 处理异常
        } finally {
            if (inputStream != null) {
                try {
                    inputStream.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
            if (outputStream != null) {
                try {
                    outputStream.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
        }
    }

2.4 excel制作下载模板

这里顺带讲一下怎么制作excel下载模板中的字典类型字段下拉限制

这样就可以实现输入限制了

三、结语

后续我还会实现附件上传,xml解析下载,pdf预览等功能,这些东西其实我觉得确实得自己留一个demo,方便自己后续需要的时候可以直接拿来用,挺好的。

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

使用maven搭建一个SpingBoot项目

1.首先创建一个maven项目 注意选择合适的jdk版本 2.添加依赖 2.在pom.xml中至少添加依赖 spring-boot-starter-web 依赖&#xff0c;目的是引入Tomcat&#xff0c;以及SpringMVC等&#xff0c;使项目具有web功能。 <!-- 引入 包含tomcat&#xff0c;SpringMVC&#xff0c…

二维Gamma分布的激光点云去噪

目录 1、Gamma 分布简介2、实现步骤 1、Gamma 分布简介 Gamma 分布在合成孔径雷达( Synthetic Aperture &#xff32;adar&#xff0c;SA&#xff32;) 图像分割中具有广泛应用&#xff0c;较好的解决了SA&#xff32; 图像中相干斑噪声对图像分割的影响。采用二维Gamma 分布对…

配置基于不同端口的虚拟主机

更改配置文件&#xff0c;添加三个不同端口的虚拟主机 <directory /www> allowoverride none require all granted </directory><virtualhost 192.168.209.136:80> documentroot /www servername 192.168.209.136 </virtualhost><virtualhost 192.…

详解yolov5的网络结构

转载自文章 网络结构图&#xff08;简易版和详细版&#xff09; 此图是博主的老师&#xff0c;杜老师的图 网络框架介绍 前言&#xff1a; YOLOv5是一种基于轻量级卷积神经网络&#xff08;CNN&#xff09;的目标检测算法&#xff0c;整体可以分为三个部分&#xff0c; ba…

Floyd判圈算法——环形链表(C++)

Floyd判圈算法(Floyd Cycle Detection Algorithm)&#xff0c;又称龟兔赛跑算法(Tortoise and Hare Algorithm)&#xff0c;是一个可以在有限状态机、迭代函数或者链表上判断是否存在环&#xff0c;求出该环的起点与长度的算法。 …

实验四 图像增强—灰度变换之直方图变换

一&#xff0e;实验目的 1&#xff0e;掌握灰度直方图的概念及其计算方法&#xff1b; 2&#xff0e;熟练掌握直方图均衡化计算过程&#xff1b;了解直方图规定化的计算过程&#xff1b; 3&#xff0e;了解色彩直方图的概念和计算方法 二&#xff0e;实验内容&#xff1a; …

【雷丰阳-谷粒商城 】【分布式高级篇-微服务架构篇】【19】认证服务03—分布式下Session共享问题

持续学习&持续更新中… 守破离 【雷丰阳-谷粒商城 】【分布式高级篇-微服务架构篇】【19】分布式下Session共享问题 session原理分布式下session共享问题Session共享问题解决—session复制Session共享问题解决—客户端存储Session共享问题解决—hash一致性Session共享问题…

嵌入式linux面试1

1. linux 1.1. Window系统和Linux系统的区别 linux区分大小写windows在dos&#xff08;磁盘操作系统&#xff09;界面命令下不区分大小写&#xff1b; 1.2. 文件格式区分 windows用扩展名区分文件&#xff1b;如.exe代表执行文件&#xff0c;.txt代表文本文件&#xff0c;.…

Seatunnel本地模式快速测验

前言 SeaTunnel&#xff08;先前称为WaterDrop&#xff09;是一个分布式、高性能、易于扩展的数据集成平台&#xff0c;旨在实现海量数据的同步和转换。它支持多种数据处理引擎&#xff0c;包括Apache Spark和Apache Flink&#xff0c;并在某个版本中引入了自主研发的Zeta引擎…

【c++】通过写一个C++函数来模拟跨境洗钱和系统警告

效果图&#xff1a; 源码&#xff1a; #include <iostream> #include <cstdlib> #include <ctime> #include <iomanip> #include <chrono> #include <thread> // 引入线程头文件#ifdef _WIN32 // 确保只在Windows上包含Windows.h #inclu…

zigbee笔记:六、看门狗定时器(Watch Dog)

一、看门狗基础 1、看门狗功能&#xff1a; 由于单片机的工作常常会受到来自外界电磁场的干扰&#xff0c;造成各种寄存器和内存的数据混乱&#xff0c;会导致程序指针错误等&#xff0c;程序运行可能会陷入死循环。程序的正常运行被打断&#xff0c;由单片机控制的系统无法继…

芯片的PPA-笔记

写在前面&#xff1a;这个仅记录自己对芯片PPA的一些思考&#xff0c;不一定正确&#xff0c;还请各位网友思辨的看待&#xff0c;欢迎大家谈谈自己的想法。 1 此次笔记的起因 记录的原因&#xff1a;自己在整理这段时间的功耗总结&#xff0c;又看到工艺对功耗的影响&#x…

12.SQL注入-盲注基于时间(base on time)

SQL注入-盲注基于时间(base on time) boolian的盲注类型还有返回信息的状态&#xff0c;但是基于时间的盲注就什么都没有返回信息。 输入payload语句进行睡5秒中&#xff0c;通过开发这工具查看时间&#xff0c;如图所示&#xff0c;会在5秒钟后在执行&#xff0c;因此存在基于…

面试篇-系统设计题总结

文章目录 1、设计一个抢红包系统1.1 高可用的解决方案&#xff1a;1.2 抢红包系统的设计1.3 其他 2、秒杀系统设计 这里记录一些有趣的系统设计类的题目&#xff0c;一般大家比较喜欢出的设计类面试题目会和高可用系统相关比如秒杀和抢红包等。欢迎大家在评论中评论自己遇到的题…

磁钢生产领域上下料解决方案

随着智能制造技术的不断革新&#xff0c;磁钢生产领域正逐步引入自动化生产线。然而&#xff0c;传统的人工上下料方式存在诸多问题&#xff0c;难以满足现代生产需求。富唯智能提出了一款复合机器人磁钢上下料解决方案&#xff0c;通过先进的自动化技术&#xff0c;提高生产效…

填报高考志愿,怎样正确地选择大学专业?

大学专业的选择&#xff0c;会关系到未来几年甚至一辈子的发展方向。这也是为什么很多人结束高考之后就开始愁眉苦脸&#xff0c;因为他们不知道应该如何选择大学专业&#xff0c;生怕一个错误的决定会影响自己一生。 毋庸置疑&#xff0c;在面对这种选择的时候&#xff0c;我…

Keycloak SSO 如何验证已添加的 SPN 是否生效

使用 Kerberos Ticket 验证&#xff1a; 在客户端计算机上&#xff0c;运行以下命令以获取 Kerberos Ticket&#xff1a; klist检查是否存在与 HTTP/yourdomain.com 相关的票证。如果存在&#xff0c;说明 SPN 已生效。 测试应用程序&#xff1a; 使用具有 HTTP/yourdomain.com…

windows USB 设备驱动开发-控制传输的数据包

每次在主机控制器和 USB 设备之间移动数据时&#xff0c;都会发生传输。 通常&#xff0c;USB 传输可大致分为控制传输和数据传输。 所有 USB 设备都必须支持控制传输&#xff0c;并且可以支持用于数据传输的端点。 每种类型的传输都与设备缓冲区USB 端点 的类型相关联。 控制传…

Linux 查看磁盘是不是 ssd 的方法

lsblk 命令检查 $ lsblk -d -o name,rota如果 ROTA 值为 1&#xff0c;则磁盘类型为 HDD&#xff0c;如果 ROTA 值为 0&#xff0c;则磁盘类型为 SSD。可以在上面的屏幕截图中看到 sda 的 ROTA 值是 1&#xff0c;表示它是 HDD。 2. 检查磁盘是否旋转 $ cat /sys/block/sda/q…

北京十大拆迁律师事务所排名

历史时刻在重演&#xff0c;土地征地拆迁作为城市发展中不可或缺的环节备受地方政府重视。然而&#xff0c;在土地征收过程中&#xff0c;往往因为拆迁补偿引发各种纠纷案件&#xff0c;给拆迁方和被拆迁方带来重大损失&#xff0c;侵害双方利益&#xff0c;尤其是被征收人。因…