Springboot开发之 Excel 处理工具(二)-- Easyexcel

一、Easyexcel 简介

EasyExcel是一个基于Java的Excel处理工具库,它的核心设计理念是快速、简洁,并且能够有效解决处理大文件时的内存溢出问题。使用EasyExcel,开发者可以在几乎不需要考虑性能和内存消耗的情况下,轻松实现Excel文件的读写操作。

  • Easyexcel 官方文档:https://easyexcel.opensource.alibaba.com/docs/current/
  • github地址:https://github.com/alibaba/easyexcel
  • gitee地址:https://gitee.com/easyexcel/easyexcel

二、为何选择 Easyexcel

1. 现有框架的问题

  • Apache POI 和 jxl: 这两个Java框架用于解析和生成Excel文件,但它们的主要问题是内存消耗大。
  • 内存溢出问题: 处理大型Excel文件时,这些框架可能会消耗大量内存,导致内存溢出。

2. Apache POI 的局限性

  • SAX模式: Apache POI 提供了SAX模式的API来部分解决内存问题,但仍然存在不足。
  • 内存消耗: 即使使用SAX模式,Apache POI 在处理Excel 2007版(.xlsx)文件时,由于解压缩和存储过程在内存中进行,内存消耗依然很大。

3. EasyExcel 的优势

  • 重写解析: EasyExcel 重写了Apache POI对Excel 2007版的解析方式,显著降低了内存消耗。
  • 内存优化: 一个3MB的Excel文件在使用Apache POI的SAX模式解析时可能需要约100MB内存,而EasyExcel可以将内存需求降低到几MB。
  • 大文件处理: 使用EasyExcel,即使更大的Excel文件也不会导致内存溢出。
  • 03版Excel处理: 对于Excel 2003版(.xls),EasyExcel依赖于Apache POI的SAX模式,并在此基础上进行了模型转换封装,使得使用更加简单和方便。

三、Springboot 项目集成 EasyExcel

1. pom 文件引入依赖:

xml 复制代码
<dependency>  
    <groupId>com.alibaba</groupId>  
    <artifactId>easyexcel</artifactId>  
    <version>3.1.0</version> <!-- 请检查并使用最新版本 -->  
</dependency>

2. 实体类(例如 UserData)

java 复制代码
import com.alibaba.excel.annotation.ExcelProperty;  
  
public class UserData {  
  
    @ExcelProperty(value = "ID", index = 0)  
    private Integer id;  
  
    @ExcelProperty(value = "姓名", index = 1)  
    private String name;  
  
    @ExcelProperty(value = "年龄", index = 2)  
    private Integer age;  
  
    // 省略 getter 和 setter 方法  
    // ...  
}

3. EasyExcel 操作帮助类

java 复制代码
import com.alibaba.excel.EasyExcel;  
import com.alibaba.excel.context.AnalysisContext;  
import com.alibaba.excel.event.AnalysisEventListener;  
import com.alibaba.excel.read.builder.ExcelReaderBuilder;  
import com.alibaba.excel.read.builder.ExcelReaderSheetBuilder;  
import com.alibaba.excel.write.builder.ExcelWriterBuilder;  
import com.alibaba.excel.write.builder.ExcelWriterSheetBuilder;  
import com.alibaba.excel.write.metadata.WriteSheet;  
  
import org.springframework.web.multipart.MultipartFile;  
  
import java.io.ByteArrayInputStream;  
import java.io.ByteArrayOutputStream;  
import java.io.IOException;  
import java.io.InputStream;  
import java.util.ArrayList;  
import java.util.List;  
  
public class EasyExcelUtils {  
  
    // 导出 Excel  
    public static byte[] exportExcel(List<?> data, String sheetName, Class<?> head) {  
        try (ByteArrayOutputStream out = new ByteArrayOutputStream()) {  
            ExcelWriterBuilder writerBuilder = EasyExcel.write(out, head);  
            ExcelWriterSheetBuilder sheetBuilder = writerBuilder.sheet(sheetName);  
            WriteSheet writeSheet = sheetBuilder.build();  
            EasyExcel.write(out, data, writeSheet).finish();  
            return out.toByteArray();  
        } catch (IOException e) {  
            throw new RuntimeException("导出 Excel 失败", e);  
        }  
    }  
  
    // 导入 Excel  
    public static <T> List<T> importExcel(MultipartFile file, Class<T> clazz) {  
        try (InputStream inputStream = file.getInputStream()) {  
            ExcelReaderBuilder readerBuilder = EasyExcel.read(inputStream, clazz, new AnalysisEventListener<T>() {  
                private List<T> list = new ArrayList<>();  
  
                @Override  
                public void invoke(T data, AnalysisContext context) {  
                    list.add(data);  
                }  
  
                @Override  
                public void doAfterAllAnalysed(AnalysisContext context) {  
                    // 数据解析完成后,可以在这里处理解析结果  
                }  
            });  
            ExcelReaderSheetBuilder sheetBuilder = readerBuilder.sheet();  
            sheetBuilder.doRead();  
            return list;  
        } catch (IOException e) {  
            throw new RuntimeException("导入 Excel 失败", e);  
        }  
    }  
  
    // 其他可能需要的帮助方法...  
}

4. 在 Controller 中使用

java 复制代码
import org.springframework.http.HttpHeaders;  
import org.springframework.http.MediaType;  
import org.springframework.http.ResponseEntity;  
import org.springframework.web.bind.annotation.GetMapping;  
import org.springframework.web.bind.annotation.PostMapping;  
import org.springframework.web.bind.annotation.RequestParam;  
import org.springframework.web.bind.annotation.RestController;  
import org.springframework.web.multipart.MultipartFile;  
  
import java.util.Arrays;  
import java.util.List;  
  
@RestController  
public class ExcelController {  
  
    @PostMapping("/exportExcel")  
    public ResponseEntity<byte[]> exportExcel() {  
        List<UserData> data = Arrays.asList(  
                new UserData(1, "Alice", 25),  
                new UserData(2, "Bob", 30),  
                new UserData(3, "Charlie", 35)
                String fileName = "users.xlsx";  
   	 	byte[] excelBytes = EasyExcelUtils.exportExcel(data, fileName, UserData.class);  

   	 	HttpHeaders headers = new HttpHeaders();  
    	headers.setContentDispositionFormData("attachment", fileName);  
    	headers.setContentType(MediaType.APPLICATION_OCTET_STREAM);  

    	return ResponseEntity.ok()  
            .headers(headers)  
            .body(excelBytes);  
	}  

	@PostMapping("/importExcel")  
	public ResponseEntity<?> importExcel(@RequestParam("file") 		MultipartFile file) {  
    	List<UserData> userDataList = EasyExcelUtils.importExcel(file, UserData.class);  

    	// 假设你需要在这里处理导入的数据,比如保存到数据库等  
    	// ...  

    	return ResponseEntity.ok("Excel 导入成功,共导入 " + userDataList.size() + " 条数据");  
}

四、前端 vue 页面调用示例

html 复制代码
<script>  
import axios from 'axios';  
  
export default {  
  // ... 其他选项和数据 ...  
  
  methods: {  
    exportExcel() {  
      // 假设你有一个要发送到服务器的对象(如果需要的话)  
      const params = {  
        // 例如: filter: 'someValue'  
      };  
  
      axios({  
        method: 'post',  
        url: '/api/exportExcel', // 修改为你的实际API地址  
        data: params, // 如果需要的话  
        responseType: 'blob', // 告诉axios我们期望返回一个blob  
      })  
      .then((response) => {  
        // 创建一个Blob对象来表示二进制数据  
        const blob = new Blob([response.data], { type: 'application/vnd.openxmlformats-officedocument.spreadsheetml.sheet' });  
        // 创建一个指向该Blob对象的URL  
        const url = window.URL.createObjectURL(blob);  
        // 创建一个a标签并模拟点击来下载文件  
        const link = document.createElement('a');  
        link.href = url;  
        link.setAttribute('download', 'users.xlsx'); // 设置下载文件名  
        document.body.appendChild(link);  
        link.click();  
        // 清理  
        window.URL.revokeObjectURL(url);  
        document.body.removeChild(link);  
      })  
      .catch((error) => {  
        console.error('导出Excel失败:', error);  
        this.message = '导出Excel失败';  
      });  
    },  
    // ... 其他方法 ...  
  },  
};  
</script>
相关推荐
!!!52533 分钟前
日志技术-LogBack入门程序&Log配置文件&日志级别
spring boot
Ai 编码助手2 小时前
在 Go 语言中如何高效地处理集合
开发语言·后端·golang
小丁爱养花2 小时前
Spring MVC:HTTP 请求的参数传递2.0
java·后端·spring
Channing Lewis2 小时前
什么是 Flask 的蓝图(Blueprint)
后端·python·flask
轩辕烨瑾3 小时前
C#语言的区块链
开发语言·后端·golang
花开盛夏^.^3 小时前
Excel常用功能总结
excel
feilieren3 小时前
SpringBoot 搭建 SSE
java·spring boot·spring
栗豆包5 小时前
w175基于springboot的图书管理系统的设计与实现
java·spring boot·后端·spring·tomcat
萧若岚6 小时前
Elixir语言的Web开发
开发语言·后端·golang
Channing Lewis6 小时前
flask实现重启后需要重新输入用户名而避免浏览器使用之前已经记录的用户名
后端·python·flask