springboot实现异步导入Excel的注意点

news2025/4/1 14:16:22

springboot实现异步导入Excel

  • 需求前言
  • 异步导入面临的问题
    • 实现异步
    • 如何导入大Excel文件避免OOM?
    • 异步操作后,如何通知导入结果?
    • 如何加快导入效率?
    • 将导入结果通知给用户后,如何避免重复通知?
  • 优化点
  • 完结撒花,如有需要收藏的看官,顺便也用发财的小手点点赞哈,如有错漏,也欢迎各位在评论区评论!

需求前言

前文介绍了使用断点续传优化同步导入Excel,即使并发上传分片,依然会因上传耗时对用户体验感不好,故上文只起到抛砖引玉的作用,生产上使用较多的还是异步导入的方式。

异步导入面临的问题

先说说流程:
1、异步导入Excel,后端使用springboot和easyExcel接收处理,并入库,同时异步导入结果会保存到消息通知表;
2、通知前端页面的收件箱有红点表示有导入结果,并且用户点击收件箱查看完,就会取消红点表示已查看过了,下次用户打开前端页面就不会有红点了;
面临的问题:
1、如何实现异步?
2、如何导入大Excel文件避免OOM?
3、异步操作后,如何通知导入结果?
4、如何加快导入效率?
5、将导入结果通知给用户后,如何避免重复通知?
往下看就知道如何解决上述问题了

实现异步

使用spring的Async注解实现异步,但要注意配置线程池,否则在并发导入时,创建多个线程处理异步容易出现OOM,代码如下:

@Configuration
@EnableAsync
public class AsyncConfig implements AsyncConfigurer {
    @Override
    public Executor getAsyncExecutor() {
        ThreadPoolTaskExecutor executor = new ThreadPoolTaskExecutor();
        executor.setCorePoolSize(5);
        executor.setMaxPoolSize(10);
        executor.setQueueCapacity(100);
        executor.setThreadNamePrefix("ExcelImport-");
        executor.initialize();
        return executor;
    }
}

@Service
@RequiredArgsConstructor
public class ExcelImportService {
    private final ImportTaskMapper taskMapper;
    private final ImportMessageMapper messageMapper;
    private final UserService userService;

    @Async
    public void asyncImport(String taskId, File excelFile) {
    	.................
    }
}

如何导入大Excel文件避免OOM?

使用阿里的easyExcel工具即可避免,前文也介绍过其操作原理,这里就不多做解释了。

异步操作后,如何通知导入结果?

在导入的大Excel文件,处理数据是异步的,所以需要将处理是成功还是失败的结果保存到一个消息通知表中,供用户访问,示例如csdn有一个消息通知的功能在这里插入图片描述

消息通知表设计:

CREATE TABLE import_message (
    id BIGINT AUTO_INCREMENT PRIMARY KEY,
    task_id VARCHAR(32) NOT NULL COMMENT '关联任务ID',
    user_id BIGINT NOT NULL COMMENT '用户ID',
    is_read TINYINT DEFAULT 0 COMMENT '是否已读(0:未读,1:已读)',
    create_time DATETIME DEFAULT CURRENT_TIMESTAMP
);

代码如下:

@Service
@RequiredArgsConstructor
public class ExcelImportService {
    private final ImportTaskMapper taskMapper;
    private final ImportMessageMapper messageMapper;
    private final UserService userService;

    @Async
    public void asyncImport(String taskId, File excelFile) {
        ImportTask task = taskMapper.selectById(taskId);
        try {
            // 使用EasyExcel解析
            ImportResult result = EasyExcel.read(excelFile)
                .head(ExcelData.class)
                .registerReadListener(new DataListener(task))
                .sheet().doRead();
            
            // 更新任务状态
            task.setStatus(1);
            task.setSuccessCount(result.getSuccessCount());
            task.setErrorCount(result.getErrorCount());
            if (result.hasErrors()) {
                task.setErrorFile(generateErrorFile(result));
            }
        } catch (Exception e) {
            task.setStatus(2);
            task.setErrorCount(-1); // 表示系统错误
        } finally {
            taskMapper.updateById(task);
            createMessage(task); // 创建通知消息
        }
    }

    private void createMessage(ImportTask task) {
        ImportMessage message = new ImportMessage();
        message.setTaskId(task.getId());
        message.setUserId(task.getUserId());
        messageMapper.insert(message);
    }
}

// 数据监听器
public class DataListener extends AnalysisEventListener<ExcelData> {
    private final ImportTask task;
    private final List<ExcelData> cachedData = new ArrayList<>();
    private final List<ErrorRow> errors = new ArrayList<>();

    @Override
    public void invoke(ExcelData data, AnalysisContext context) {
        // 数据校验逻辑...
        cachedData.add(data);
        if (cachedData.size() >= 100) {
            saveBatch(cachedData);
            cachedData.clear();
        }
    }

    @Override
    public void doAfterAllAnalysed(AnalysisContext context) {
        if (!cachedData.isEmpty()) {
            saveBatch(cachedData);
        }
    }
    
    private void saveBatch(List<ExcelData> list) {
        try {
            // 批量入库逻辑
        } catch (Exception e) {
            // 记录错误行
            errors.add(new ErrorRow(/* 行信息 */));
        }
    }
}

@RestController
@RequestMapping("/api/import")
@RequiredArgsConstructor
public class ImportController {
    private final ExcelImportService importService;
    
    // 获取未读消息数量
    @GetMapping("/unread-count")
    public ResponseEntity<Integer> getUnreadCount(@AuthenticationPrincipal User user) {
        int count = messageMapper.countUnread(user.getId());
        return ResponseEntity.ok(count);
    }

    // 标记消息已读
    @PostMapping("/mark-read")
    public ResponseEntity<?> markAsRead(@RequestBody List<Long> messageIds) {
        messageMapper.updateReadStatus(messageIds, 1);
        return ResponseEntity.ok().build();
    }

    // 启动异步导入
    @PostMapping
    public ResponseEntity<?> startImport(@RequestParam MultipartFile file) {
        String taskId = IdUtil.simpleUUID();
        File tempFile = saveToTemp(file); // 保存临时文件
        
        ImportTask task = new ImportTask();
        task.setId(taskId);
        task.setUserId(SecurityUtils.getCurrentUserId());
        task.setFileName(file.getOriginalFilename());
        task.setStatus(0);
        taskMapper.insert(task);

        importService.asyncImport(taskId, tempFile);
        return ResponseEntity.ok(Map.of("taskId", taskId));
    }
}

如何加快导入效率?

避免for循环每次与DB建立一个连接,应该使用MyBatis-Plus的批量插入

List<User> userList = new ArrayList<>();
User user;
for(int i = 0 ;i < 10000; i++) {
    user = new User();
    user.setUsername("name" + i);
    user.setPassword("password" + i);
    userList.add(user);
}
saveBatch(userList);

MyBatis-Plus的saveBatch方法默认是使用JDBC的addBatch()和executeBatch()方法实现批量插入。但是部分数据库的JDBC驱动并不支持addBatch(),这样每次插入都会发送一条SQL语句,严重影响了批量插入的性能。设置rewriteBatchedStatements=true后,MyBatis-Plus会重写插入语句,将其合并为一条SQL语句,从而减少网络交互次数,提高批量插入的效率。

将导入结果通知给用户后,如何避免重复通知?

前面也说过,导入Excel的结果会保存到消息表中,前端在登录后,通过访问“标记消息已读”接口标记为已读,至于前端如何发现有消息结果,直接使用定时轮训即可(csdn也是用这种来发现消息通知),前端访问代码如下:

<template>
  <!-- 上传组件 -->
  <el-upload
    action="/api/import"
    :show-file-list="false"
    :before-upload="beforeUpload"
    @success="handleSuccess"
  >
    <el-button type="primary">导入Excel</el-button>
  </el-upload>

  <!-- 通知红点 -->
  <el-badge :value="unreadCount" :max="99" class="notification-badge">
    <el-button icon="bell" @click="showMessages"></el-button>
  </el-badge>

  <!-- 消息弹窗 -->
  <el-dialog v-model="messageVisible">
    <el-table :data="messages">
      <el-table-column prop="fileName" label="文件名"></el-table-column>
      <el-table-column prop="status" label="状态">
        <template #default="{row}">
          <el-tag :type="statusType(row)">{{ statusText(row) }}</el-tag>
        </template>
      </el-table-column>
      <el-table-column label="操作">
        <template #default="{row}">
          <el-button @click="downloadError(row)" v-if="row.errorFile">
            下载错误报告
          </el-button>
        </template>
      </el-table-column>
    </el-table>
  </el-dialog>
</template>

<script setup>
import { ref, onMounted } from 'vue'
import { ElMessage } from 'element-plus'

const unreadCount = ref(0)
const messageVisible = ref(false)
const messages = ref([])

// 初始化获取未读数量
onMounted(async () => {
  const res = await fetch('/api/import/unread-count')
  unreadCount.value = await res.json()
})

// 显示消息弹窗
const showMessages = async () => {
  const res = await fetch('/api/import/messages')
  messages.value = await res.json()
  messageVisible.value = true
  
  // 标记所有消息为已读
  const ids = messages.value.map(m => m.id)
  await fetch('/api/import/mark-read', {
    method: 'POST',
    body: JSON.stringify(ids)
  })
  unreadCount.value = 0
}

// 定时刷新未读数量
setInterval(async () => {
  const res = await fetch('/api/import/unread-count')
  unreadCount.value = await res.json()
}, 30000)
</script>

优化点

1、通知导入Excel的结果,如果是导入失败,需要知道是什么原因,例如是校验某一行数据的参数不合法之类的,那也要知道是哪一行数据才行,可能会有多行数据有问题,可以通过导出一个Excel的方式,生成导出的Excel路径保存到消息通知表,前端查看红点收件箱即可下载;
2、前端将定时访问优化成websocket,避免长时间轮训,浪费带宽,当然这种优化是看业务场景是否需要,如果需要频繁导入Excel频繁通知导入结果的场景;

完结撒花,如有需要收藏的看官,顺便也用发财的小手点点赞哈,如有错漏,也欢迎各位在评论区评论!

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

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

相关文章

Linux练习——有关硬盘、联网、软件包的管理

1、将你的虚拟机的网卡模式设置为nat模式&#xff0c;给虚拟机网卡配置三个主机位分别为100、200、168的ip地址 #使用nmtui打开文本图形界面配置网络 [rootrhcsa0306 ~]# nmtui #使用命令激活名为 ens160 的 NetworkManager 网络连接 [rootrhcsa0306 ~]# nmcli c up ens160 #通…

论文阅读:GS-Blur: A 3D Scene-Based Dataset for Realistic Image Deblurring

今天介绍一篇 2024 NeurIPS 的文章&#xff0c;是关于真实世界去模糊任务的数据集构建的工作&#xff0c;论文作者来自韩国首尔大学 Abstract 要训练去模糊网络&#xff0c;拥有一个包含成对模糊图像和清晰图像的合适数据集至关重要。现有的数据集收集模糊图像的方式主要有两…

Cocos Creator Shader入门实战(七):RGB不同算法效果的实现,及渲染技术、宏定义、属性参数的延伸配置

引擎&#xff1a;3.8.5 您好&#xff0c;我是鹤九日&#xff01; 回顾 上篇文章&#xff0c;讲解了Cocos Shader如何通过setProperty动态设置材质的属性&#xff0c;以及设置属性时候的一些注意事项&#xff0c;比如&#xff1a; 一、CCEffect部分properties参数的设定后&…

算法学习记录:递归

递归算法的关键在于回复现场&#xff0c;dfs&#xff08;&#xff09;函数返回值、结束条件、它的作用。 目录 1.综合练习 2. 二叉树的深搜 1.综合练习 39. 组合总和 - 力扣&#xff08;LeetCode&#xff09; 关键在画出的决策树当中&#xff0c;前面使用过的2、3&#xff0c;…

可发1区的超级创新思路(python\matlab实现):MPTS+Lconv+注意力集成机制的Transformer时间序列模型

首先声明,该模型为原创!原创!原创!且该思路还未有成果发表,感兴趣的小伙伴可以借鉴! 应用场景 该模型主要用于时间序列数据预测问题,包含功率预测、电池寿命预测、电机故障检测等等。 一、模型整体架构(本文以光伏功率预测为例) 本模型由多尺度特征提取模块(MPTS)…

三、分类模块,通用组件顶部导航栏Navbar

1.封装通用组件顶部导航栏Navbar 不同效果 Component export struct MkNavbar {Prop title: string Prop leftIcon: ResourceStr $r("app.media.ic_public_left")ProprightIcon: ResourceStr $r("app.media.ic_public_more")PropshowLeftIcon: boolean…

PHY——LAN8720A 寄存器读写 (二)

文章目录 PHY——LAN8720A 寄存器读写 (二)工程配置引脚初始化代码以太网初始化代码PHY 接口实现LAN8720 接口实现PHY 接口测试 PHY——LAN8720A 寄存器读写 (二) 工程配置 这里以野火电子的 F429 开发板为例&#xff0c;配置以太网外设 这里有一点需要注意原理图 RMII_TXD0…

Flutter_学习记录_AppBar中取消leading的占位展示

将leading设置为null将automaticallyImplyLeading设置为false 看看automaticallyImplyLeading的说明&#xff1a; Controls whether we should try to imply the leading widget if null. If true and [AppBar.leading] is null, automatically try to deduce what the leading…

未来派几何风格包装徽标品牌海报标牌logo设计无衬线英文字体安装包 Myfonts – Trakya Sans Font Family

Trakya Sans 是一种具有几何风格的现代无衬线字体。Futura、Avant Garde 等。它具有现代条纹&#xff0c;这是宽度和高度协调的结果&#xff0c;尤其是在小写字母中&#xff0c;以支持易读性。 非常适合广告和包装、编辑和出版、徽标、品牌和创意产业、海报和广告牌、小文本、寻…

C语言深度解析:从零到系统级开发的完整指南

一、C语言的核心特性与优势 1. 高效性与直接硬件控制 C语言通过编译为机器码的特性&#xff0c;成为系统级开发的首选语言。例如&#xff0c;Linux内核通过C语言直接操作内存和硬件寄存器&#xff0c;实现高效进程调度。 关键点&#xff1a; malloc/free直接管理内存&#…

ctfshow WEB web8

首先确定注入点&#xff0c;输入以下payload使SQL恒成立 ?id-1/**/or/**/true 再输入一下payload 使SQL恒不成立 ?id-1/**/or/**/false 由于SQL恒不成立, 数据库查询不到任何数据, 从而导致页面空显示 由以上返回结果可知&#xff0c;该页面存在SQL注入&#xff0c;注入点…

【Linux】U-Boot 加载并启动 Linux 系统程序

U-Boot 加载并启动 Linux 系统程序 零、介绍 最近在玩一些嵌入式的开发板&#xff0c;在引导操作系统时需要用到U-Boot&#xff0c;故此研究一下。 U-Boot&#xff08;Universal Bootloader&#xff09;是一款开源的通用引导加载程序&#xff0c;专为嵌入式系统设计&#xff…

jarvisoj API调用 [JSON格式变XXE]

http://web.jarvisoj.com:9882/ 题目要求&#xff1a;请设法获得目标机器 /home/ctf/flag.txt 中的flag值 抓包得到&#xff1a; POST /api/v1.0/try HTTP/1.1 Host: web.jarvisoj.com:9882 Content-Length: 36 Accept-Language: zh-CN,zh;q0.9 User-Agent: Mozilla/5.0 (W…

机器学习的一百个概念(4)下采样

前言 本文隶属于专栏《机器学习的一百个概念》&#xff0c;该专栏为笔者原创&#xff0c;引用请注明来源&#xff0c;不足和错误之处请在评论区帮忙指出&#xff0c;谢谢&#xff01; 本专栏目录结构和参考文献请见[《机器学习的一百个概念》 ima 知识库 知识库广场搜索&…

NNI 适配 TensorRT10教程

引言 本文涉及两个框架及其版本分别为 NNI (Neural Network Intelligence) &#xff1a;3.0TensorRT&#xff1a;10.9.0.34 NNI 在文档 Speed Up Quantized Model with TensorRT里描述了如何使用 TensorRT 为NNI量化的模型实现加速&#xff0c;但是从NNI 的源代码https://gi…

多路径 TCP 调度的另一面

参考前面的文章 一个原教旨的多路径 TCP 和 MP-BBR 公平性推演&#xff0c;一直都破而不立&#xff0c;不能光说怎样不好&#xff0c;还得说说现状情况下&#xff0c;该如何是好。 如果 receiver 乱序重排的能力有限(拜 TCP 所赐)&#xff0c;如果非要在多路径上传输 TCP&…

vcpkg安装指定版本的库

一.vcpkg安装 使用git将vcpkg源码克隆到本地制定目录&#xff08;D:\vcpkg&#xff09;&#xff0c;并初始化 git clone https://github.com/microsoft/vcpkg.git cd vcpkg ./bootstrap-vcpkg.sh # Linux/macOS .\bootstrap-vcpkg.bat # Windows 如下图&#xff1a; 二.安…

【工具变量】上市公司供应链稳定性数据两个维度(2013-2023年)

供应链稳定性是指供应链在面对各种内外部因素的冲击和不确定性时&#xff0c;能够保持持续、顺畅运作的能力&#xff0c;而供应链稳定性指数是用于评估企业在其供应链管理中保持稳定性的一个重要指标。本分享数据参考钟涛&#xff08;2022&#xff09;、董浩和闫晴&#xff08;…

Redis场景问题2:缓存击穿

Redis 缓存击穿是指在缓存系统中&#xff0c;大量请求&#xff08;高并发访问&#xff09;同时访问一个不存在于缓存中&#xff08;一般是因为缓存过期或者数据未被加载到缓存&#xff09;但在数据库中存在的热点数据&#xff0c;从而导致这些请求直接穿透缓存层&#xff0c;涌…

RocketMQ - 从消息可靠传输谈高可用

先稍微介绍下RocketMQ架构。 主从架构 Broker 集群&#xff1a;每个 Broker 分为 Master 和 Slave 角色&#xff0c;Master 负责读写&#xff0c;Slave 作为热备。 同步复制&#xff08;SYNC_MASTER&#xff09;&#xff1a;消息写入 Master 后&#xff0c;需等待 Slave 同步完…