JAVA读写Excel(jxl,poi,easyExcel)

2024-09-04 13:12
文章标签 java excel 读写 easyexcel poi jxl

本文主要是介绍JAVA读写Excel(jxl,poi,easyExcel),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

目录

一、需求描述

二、具体操作Excel的常用方法

方法一: 使用jxl

方法二: POI

方法三:EasyExcel

三、总结


一、需求描述

前端有时候会传送 Excel 文件给后端(Java)去解析,那我们作为后端该如何实现对 Excel 文件的解析和数据读取呢?

使用Java对Excel文件进行读写,文件后缀可能为.xls, .xlsx

二、具体操作Excel的常用方法

方法一: 使用jxl

JXL:  开源库,支持.xls文件(EXCEL 2003格式)的读写

注意: JXL不支持.xlsx文件(EXCEL 2007格式)的读写

使用:

1. pom.xml文件引入 <!--读取excel文件--><!-- https://mvnrepository.com/artifact/net.sourceforge.jexcelapi/jxl --><dependency><groupId>net.sourceforge.jexcelapi</groupId><artifactId>jxl</artifactId><version>2.6.12</version></dependency>2. jxl读文件import jxl.Cell;
import jxl.Sheet;
import jxl.Workbook;
import jxl.write.Label;
import jxl.write.WritableSheet;
import jxl.write.WritableWorkbook;
import jxl.write.WriteException;import java.io.File;
import java.io.IOException;
import java.util.ArrayList;//读文件/*** 返回上传的Excel表格的内容* Jxl是一款常用的Java中操作Excel的API,但其只对xls有效,对2007版本以上的Excel(xlsx)很难处理。*/public static List<String[]> parseExcel(String filePath) throws Exception {List<String[]> list = new ArrayList<>();Workbook wb = Workbook.getWorkbook(new File(filePath));Sheet sheet = wb.getSheets()[0];int columns = sheet.getRow(0).length;for (int i = 0; i < sheet.getRows(); i++) {String[] line = new String[columns];for (int j = 0; j < columns; j++) {Cell cell = sheet.getCell(j, i);String content = null;if (cell != null) {content = cell.getContents();}if (content != null && content.trim().length() == 0) {content = "";}line[j] = content;}list.add(line);}return list;}// 写文件public static void write2File(String filePath, List<GradeVo> gradeVoList) throws IOException, WriteException {// 写入文件File file = new File(filePath);// 创建工作簿WritableWorkbook workbook = Workbook.createWorkbook(file);// 创建工作表WritableSheet sheet = workbook.createSheet("Sheet1", 0);AtomicInteger rowIndex = new AtomicInteger(1);try {//写表头Label label0 = new Label(0, 0, "学号");sheet.addCell(label0);label0 = new Label(1, 0, "姓名");sheet.addCell(label0);label0 = new Label(2, 0, "班级");sheet.addCell(label0);label0 = new Label(3, 0, "每次作业分");sheet.addCell(label0);label0 = new Label(4, 0, "作业平均分");sheet.addCell(label0);label0 = new Label(5, 0, "考勤次数");sheet.addCell(label0);label0 = new Label(6, 0, "回答问题次数");sheet.addCell(label0);label0 = new Label(7, 0, "期末分数");sheet.addCell(label0);label0 = new Label(8, 0, "最终分");sheet.addCell(label0);// 创建单元格并写入数据for (GradeVo value : gradeVoList) {Label label = new Label(0, rowIndex.get(), value.getSNo());sheet.addCell(label);label = new Label(1, rowIndex.get(), value.getSName());sheet.addCell(label);label = new Label(2, rowIndex.get(), value.getSClass());sheet.addCell(label);label = new Label(3, rowIndex.get(), JSON.toJSONString(value.getGrade()));sheet.addCell(label);label = new Label(4, rowIndex.get(), value.getAvgScore() + "");sheet.addCell(label);label = new Label(5, rowIndex.get(), value.getSignInTimes() + "");sheet.addCell(label);label = new Label(6, rowIndex.get(), value.getAnswerTimes() + "");sheet.addCell(label);label = new Label(7, rowIndex.get(), value.getFinalScore() + "");sheet.addCell(label);label = new Label(8, rowIndex.get(), value.getTotalScore() + "");sheet.addCell(label);rowIndex.addAndGet(1);}// 写入数据workbook.write();// 关闭工作簿workbook.close();}catch (WriteException e) {throw new RuntimeException(e);}}

更多用法:

//单元格合并
sheet.mergeCells(0,2,0,3); //把第 0 列第 2 行的单元格和第 0 列第 3 行的单元格合并//设置单元格列宽和行高
sheet.setColumnView(0,10);//设置第 0 列的宽为10
sheet.setRowView(0,1000);//设置第 0 行的高为10//插入字符串数据可用String[]数组(注意需要在下面遍历数组!)
String[] str = {"1","2"};//设置单元格内容样式                           设置字体              字体大小   是否加粗
WritableFont font=new WritableFont(WritableFont.createFont("宋体"),12,WritableFont.BOLD);
WritableCellFormat format = new WritableCellFormat(font);
//水平居中
format.setAlignment(jxl.format.Alignment.CENTRE);

方法二: POI

POI (Poor Obfuscation Implementation)主要是指 Apache提供的操作Office的基础工具包(Apache POI),Apache POI 是一个底层的工具类,可实现的功能最全。

缺点:

解析文件时一次性全部加载到内存中,系统并发量不大时可行,一旦并发上来后会产生OOM或者JVM频繁的full gc

操作:

pom文件引入:

引入 Apache POI 和 Apache POI-OOXML 这两个类库,Maven坐标如下:

<!-- Apache POI -->
<dependency><groupId>org.apache.poi</groupId><artifactId>poi</artifactId><version>5.2.3</version>
</dependency>
<dependency><groupId>org.apache.poi</groupId><artifactId>poi-ooxml</artifactId><version>5.2.3</version>
</dependency>
注意:使用HSSF操作.xls格式文件,使用XSSF操作.xlsx文件 读文件:private final static String xls = "xls";private final static String xlsx = "xlsx";/*** 读入excel文件,解析后返回* @param file* @throws IOException*/public static List<String[]> readExcel(MultipartFile file) throws IOException {//检查文件checkFile(file);//获得Workbook工作薄对象Workbook workbook = getWorkBook(file);//创建返回对象,把每行中的值作为一个数组,所有行作为一个集合返回List<String[]> list = new ArrayList<String[]>();if(workbook != null){for(int sheetNum = 0;sheetNum < workbook.getNumberOfSheets();sheetNum++){//获得当前sheet工作表Sheet sheet = workbook.getSheetAt(sheetNum);if(sheet == null){continue;}//获得当前sheet的开始行int firstRowNum  = sheet.getFirstRowNum();//获得当前sheet的结束行int lastRowNum = sheet.getLastRowNum();//循环除了第一行的所有行for(int rowNum = firstRowNum+1;rowNum <= lastRowNum;rowNum++){ //为了过滤到第一行因为我的第一行是数据库的列//获得当前行Row row = sheet.getRow(rowNum);if(row == null){continue;}//获得当前行的开始列int firstCellNum = row.getFirstCellNum();//获得当前行的列数int lastCellNum = row.getLastCellNum();//为空列获取
//                    int lastCellNum = row.getPhysicalNumberOfCells();//为空列不获取
//                    String[] cells = new String[row.getPhysicalNumberOfCells()];String[] cells = new String[row.getLastCellNum()];//循环当前行for(int cellNum = firstCellNum; cellNum < lastCellNum;cellNum++){Cell cell = row.getCell(cellNum);cells[cellNum] = getCellValue(cell);}list.add(cells);}}}logger.info(gson.toJson(list));return list;}public static void checkFile(MultipartFile file) throws IOException{//判断文件是否存在if(null == file){throw new FileNotFoundException("文件不存在!");}//获得文件名String fileName = file.getOriginalFilename();//判断文件是否是excel文件if(!fileName.endsWith(xls) && !fileName.endsWith(xlsx)){throw new IOException(fileName + "不是excel文件");}}public static Workbook getWorkBook(MultipartFile file) {//获得文件名String fileName = file.getOriginalFilename();//创建Workbook工作薄对象,表示整个excelWorkbook workbook = null;try {//获取excel文件的io流InputStream is = file.getInputStream();//根据文件后缀名不同(xls和xlsx)获得不同的Workbook实现类对象if(fileName.endsWith(xls)){//2003workbook = new HSSFWorkbook(is);}else if(fileName.endsWith(xlsx)){//2007workbook = new XSSFWorkbook(is);}} catch (IOException e) {logger.info(e.getMessage());}return workbook;}public static String getCellValue(Cell cell){String cellValue = "";if(cell == null){return cellValue;}//把数字当成String来读,避免出现1读成1.0的情况if(cell.getCellType() == Cell.CELL_TYPE_NUMERIC){cell.setCellType(Cell.CELL_TYPE_STRING);}//判断数据的类型switch (cell.getCellType()){case Cell.CELL_TYPE_NUMERIC: //数字cellValue = String.valueOf(cell.getNumericCellValue());break;case Cell.CELL_TYPE_STRING: //字符串cellValue = String.valueOf(cell.getStringCellValue());break;case Cell.CELL_TYPE_BOOLEAN: //BooleancellValue = String.valueOf(cell.getBooleanCellValue());break;case Cell.CELL_TYPE_FORMULA: //公式
//                cellValue = String.valueOf(cell.getCellFormula());cellValue = String.valueOf(cell.getStringCellValue());break;case Cell.CELL_TYPE_BLANK: //空值cellValue = "";break;case Cell.CELL_TYPE_ERROR: //故障cellValue = "非法字符";break;default:cellValue = "未知类型";break;}return cellValue;}

官网文档:

Busy Developers' Guide to HSSF and XSSF Features

  • 工作簿对象支持设置一些全局性质的对象,例如:
  1.  单元格样式的对象。一个工作簿最多支持6w个单元格的样式;
  2.  创建字体样式的对象。一个工作簿最多支持32767个 字体样式;
  • 工作表对象支持只在单个工作表生效的设置,例如:
  1. 设置筛选按钮; 
  2. 设置某列的自适应列宽;
  3. 冻结窗格;
  4. 拆分窗口;
  • 行对象

Row 支持的操作有:

  1. 创建单元格
  2. 获取/遍历单元格
  3. 设置行高
  4. 获取行号
  5. 设置/隐藏行
  • 单元格对象

单元格对象支持对单个单元格进行设置,例如:

  1. 获取/设置单元格的值
  2. 获取/设置单元格的类型(字符串、数值[数字/日期]、布尔值、公式、错误值、空)
  3. 获取/设置单元格的样式
  4. 获取/设置超链接
  • 单元格的值

    单元格的值有7种类型[CellType](数值、字符串、公式、空值、布尔值、错误值)

方法三:EasyExcel

优点:

Java解析、生成Excel比较有名的框架有Apache poi、jxl。但他们都存在一个严重的问题就是非常的耗内存,poi有一套SAX模式的API可以一定程度的解决一些内存溢出的问题,但POI还是有一些缺陷,比如07版Excel解压缩以及解压后存储都是在内存中完成的,内存消耗依然很大。
        easyexcel重写了poi对07版Excel的解析,一个3M的excel用POI sax解析依然需要100M左右内存,改用easyexcel可以降低到几M,并且再大的excel也不会出现内存溢出;03版依赖POI的sax模式,在上层做了模型转换的封装,让使用者更加简单方便

操作:

pom文件:

读Excel:DEMO代码地址:https://github.com/alibaba/easyexcel/blob/master/easyexcel-test/src/test/java/com/alibaba/easyexcel/test/demo/read/ReadTest.java
/*** 最简单的读* <p>1. 创建excel对应的实体对象 参照{@link DemoData}* <p>2. 由于默认一行行的读取excel,所以需要创建excel一行一行的回调监听器,参照{@link DemoDataListener}* <p>3. 直接读即可*/@Testpublic void simpleRead() {String fileName = TestFileUtil.getPath() + "demo" + File.separator + "demo.xlsx";// 这里 需要指定读用哪个class去读,然后读取第一个sheet 文件流会自动关闭EasyExcel.read(fileName, DemoData.class, new DemoDataListener()).sheet().doRead();}写Excel:DEMO代码地址:https://github.com/alibaba/easyexcel/blob/master/easyexcel-test/src/test/java/com/alibaba/easyexcel/test/demo/write/WriteTest.java
/*** 最简单的写* <p>1. 创建excel对应的实体对象 参照{@link com.alibaba.easyexcel.test.demo.write.DemoData}* <p>2. 直接写即可*/@Testpublic void simpleWrite() {String fileName = TestFileUtil.getPath() + "write" + System.currentTimeMillis() + ".xlsx";// 这里 需要指定写用哪个class去读,然后写到第一个sheet,名字为模板 然后文件流会自动关闭// 如果这里想使用03 则 传入excelType参数即可EasyExcel.write(fileName, DemoData.class).sheet("模板").doWrite(data());}

官网文档:

https://easyexcel.opensource.alibaba.com/docs/current/EasyExcel是一个基于Java的简单、省内存的读写Excel的开源项目,在尽可能节约内存的情况下支持读写百M的Excel。icon-default.png?t=N7T8https://easyexcel.opensource.alibaba.com/docs/current/读Excel | Easy Excel 官网

三、总结

  • 如果操作Excel复杂度高(.xls,.xlsx都存在,且内容格式复杂),建议使用POI。
  • 如果操作Excel数据量大并且对对性能有要求,可以使用EasyExcel。

这篇关于JAVA读写Excel(jxl,poi,easyExcel)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!


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

相关文章

Spring Boot 整合 SSE的高级实践(Server-Sent Events)

《SpringBoot整合SSE的高级实践(Server-SentEvents)》SSE(Server-SentEvents)是一种基于HTTP协议的单向通信机制,允许服务器向浏览器持续发送实... 目录1、简述2、Spring Boot 中的SSE实现2.1 添加依赖2.2 实现后端接口2.3 配置超时时

Spring Boot读取配置文件的五种方式小结

《SpringBoot读取配置文件的五种方式小结》SpringBoot提供了灵活多样的方式来读取配置文件,这篇文章为大家介绍了5种常见的读取方式,文中的示例代码简洁易懂,大家可以根据自己的需要进... 目录1. 配置文件位置与加载顺序2. 读取配置文件的方式汇总方式一:使用 @Value 注解读取配置方式二

一文详解Java异常处理你都了解哪些知识

《一文详解Java异常处理你都了解哪些知识》:本文主要介绍Java异常处理的相关资料,包括异常的分类、捕获和处理异常的语法、常见的异常类型以及自定义异常的实现,文中通过代码介绍的非常详细,需要的朋... 目录前言一、什么是异常二、异常的分类2.1 受检异常2.2 非受检异常三、异常处理的语法3.1 try-

Java中的@SneakyThrows注解用法详解

《Java中的@SneakyThrows注解用法详解》:本文主要介绍Java中的@SneakyThrows注解用法的相关资料,Lombok的@SneakyThrows注解简化了Java方法中的异常... 目录前言一、@SneakyThrows 简介1.1 什么是 Lombok?二、@SneakyThrows

Java中字符串转时间与时间转字符串的操作详解

《Java中字符串转时间与时间转字符串的操作详解》Java的java.time包提供了强大的日期和时间处理功能,通过DateTimeFormatter可以轻松地在日期时间对象和字符串之间进行转换,下面... 目录一、字符串转时间(一)使用预定义格式(二)自定义格式二、时间转字符串(一)使用预定义格式(二)自

Spring 请求之传递 JSON 数据的操作方法

《Spring请求之传递JSON数据的操作方法》JSON就是一种数据格式,有自己的格式和语法,使用文本表示一个对象或数组的信息,因此JSON本质是字符串,主要负责在不同的语言中数据传递和交换,这... 目录jsON 概念JSON 语法JSON 的语法JSON 的两种结构JSON 字符串和 Java 对象互转

JAVA保证HashMap线程安全的几种方式

《JAVA保证HashMap线程安全的几种方式》HashMap是线程不安全的,这意味着如果多个线程并发地访问和修改同一个HashMap实例,可能会导致数据不一致和其他线程安全问题,本文主要介绍了JAV... 目录1. 使用 Collections.synchronizedMap2. 使用 Concurren

Java Response返回值的最佳处理方案

《JavaResponse返回值的最佳处理方案》在开发Web应用程序时,我们经常需要通过HTTP请求从服务器获取响应数据,这些数据可以是JSON、XML、甚至是文件,本篇文章将详细解析Java中处理... 目录摘要概述核心问题:关键技术点:源码解析示例 1:使用HttpURLConnection获取Resp

Java的栈与队列实现代码解析

《Java的栈与队列实现代码解析》栈是常见的线性数据结构,栈的特点是以先进后出的形式,后进先出,先进后出,分为栈底和栈顶,栈应用于内存的分配,表达式求值,存储临时的数据和方法的调用等,本文给大家介绍J... 目录栈的概念(Stack)栈的实现代码队列(Queue)模拟实现队列(双链表实现)循环队列(循环数组

Java中Switch Case多个条件处理方法举例

《Java中SwitchCase多个条件处理方法举例》Java中switch语句用于根据变量值执行不同代码块,适用于多个条件的处理,:本文主要介绍Java中SwitchCase多个条件处理的相... 目录前言基本语法处理多个条件示例1:合并相同代码的多个case示例2:通过字符串合并多个case进阶用法使用