easyexcel和poi对比_POI和EasyExcel的使用

1.POI使用

1.1 什么是POI

POI简介(Apache POI),Apache POI是Apache软件基金会的开放源码函式库,POI提供API给Java程序对Microsoft Office格式档案读和写的功能。

HSSF - 提供读写Microsoft Excel格式档案的功能。(.xls)

XSSF - 提供读写Microsoft Excel OOXML格式档案的功能。(.xlsx)

HWPF - 提供读写Microsoft Word格式档案的功能。

HSLF - 提供读写Microsoft PowerPoint格式档案的功能。

HDGF - 提供读写Microsoft Visio格式档案的功能。

1.2 官网

官网可以找到文档和每个版本的下载地址

2.xls写-03和07

2.1、创建项目

1、创建一个普通的maven项目

项目名:excel_poi

2、pom中引入xml相关依赖

org.apache.poi

poi

3.9

org.apache.poi

poi-ooxml

3.9

joda-time

joda-time

2.10.1

junit

junit

4.12

2.2、xls写-03和07的区别

区别1.格式不一样,03是xls,07是xlsx。

区别2:别的没啥区别。

1、03

public class ExcelWriteTest {

@Test

public void testWrite03() throws IOException {

// 创建新的Excel 工作簿

Workbook workbook = new HSSFWorkbook();

// 在Excel工作簿中建一工作表,其名为缺省值 Sheet0

//Sheet sheet = workbook.createSheet();

// 如要新建一名为"会员登录统计"的工作表,其语句为:

Sheet sheet = workbook.createSheet("会员登录统计");

// 创建行(row 1)

Row row1 = sheet.createRow(0);

// 创建单元格(col 1-1)

Cell cell11 = row1.createCell(0);

cell11.setCellValue("今日人数");

// 创建单元格(col 1-2)

Cell cell12 = row1.createCell(1);

cell12.setCellValue(666);

// 创建行(row 2)

Row row2 = sheet.createRow(1);

// 创建单元格(col 2-1)

Cell cell21 = row2.createCell(0);

cell21.setCellValue("统计时间");

//创建单元格(第三列)

Cell cell22 = row2.createCell(1);

String dateTime = new DateTime().toString("yyyy-MM-dd HH:mm:ss");

cell22.setCellValue(dateTime);

// 新建一输出文件流(注意:要先创建文件夹)

FileOutputStream out = new FileOutputStream("d:/excel-poi/test-write03.xls");

// 把相应的Excel 工作簿存盘

workbook.write(out);

// 操作结束,关闭文件

out.close();

System.out.println("文件生成成功");

}

}

2 、07

@Test

public void testWrite07() throws IOException {

// 创建新的Excel 工作簿

Workbook workbook = new XSSFWorkbook();

......

// 新建一输出文件流(注意:要先创建文件夹)

FileOutputStream out = new FileOutputStream("d:/excel-poi/test-write07.xlsx");

......

}

2.3、大文件写-HSSF、XSSF和SXSSF

1、使用HSSF

缺点:最多只能处理65536行,否则会抛出异常

java.lang.IllegalArgumentException: Invalid row number (65536) outside allowable range (0..65535)

优点:过程中写入缓存,不操作磁盘,最后一次性写入磁盘,速度快

@Test

public void testWrite03BigData() throws IOException {

//记录开始时间

long begin = System.currentTimeMillis();

//创建一个SXSSFWorkbook

Workbook workbook = new HSSFWorkbook();

//创建一个sheet

Sheet sheet = workbook.createSheet();

//xls文件最大支持65536行

for (int rowNum = 0; rowNum < 65536; rowNum++) {

//创建一个行

Row row = sheet.createRow(rowNum);

for (int cellNum = 0; cellNum < 10; cellNum++) {//创建单元格

Cell cell = row.createCell(cellNum);

cell.setCellValue(cellNum);

}

}

System.out.println("done");

FileOutputStream out = new FileOutputStream("d:/excel-poi/test-write03-bigdata.xls");

workbook.write(out);

// 操作结束,关闭文件

out.close();

//记录结束时间

long end = System.currentTimeMillis();

System.out.println((double)(end - begin)/1000);

}

2、使用XSSF

缺点:写数据时速度非常慢,非常耗内存,也会发生内存溢出,如100万条

优点:可以写较大的数据量,如20万条

@Test

public void testWrite07BigData() throws IOException {

//记录开始时间

long begin = System.currentTimeMillis();

//创建一个XSSFWorkbook

Workbook workbook = new XSSFWorkbook();

......

FileOutputStream out = new FileOutputStream("d:/excel-poi/test-write07-bigdata.xlsx");

......

}

3、使用SXSSF

优点:可以写非常大的数据量,如100万条甚至更多条,写数据速度快,占用更少的内存

注意:

过程中会产生临时文件,需要清理临时文件

默认由100条记录被保存在内存中,如果查过这数量,则最前面的数据被写入临时文件

如果想自定义内存中数据的数量,可以使用new SXSSFWorkbook(数量)

@Test

public void testWrite07BigDataFast() throws IOException {

//记录开始时间

long begin = System.currentTimeMillis();

//创建一个SXSSFWorkbook

Workbook workbook = new SXSSFWorkbook();

......

FileOutputStream out = new FileOutputStream("d:/excel-poi/test-write07-bigdata-fast.xlsx");

workbook.write(out);

// 操作结束,关闭文件

out.close();

//清除临时文件

((SXSSFWorkbook)workbook).dispose();

//记录结束时间

long end = System.currentTimeMillis();

System.out.println((double)(end - begin)/1000);

}

SXSSFWorkbook-来至官方的解释:实现“BigGridDemo”策略的流式XSSFWorkbook版本。这允许写入非常大的文件而不会耗尽内存,因为任何时候只有可配置的行部分被保存在内存中。

请注意,仍然可能会消耗大量内存,这些内存基于您正在使用的功能,例如合并区域,注释......仍然只存储在内存中,因此如果广泛使用,可能需要大量内存。

3、xls读-03和07

1、03

package com.atguigu.excelpoi;

public class ExcelReadTest {

@Test

public void testRead03() throws Exception{

InputStream is = new FileInputStream("d:/excel-poi/商品表-03.xls");

Workbook workbook = new HSSFWorkbook(is);

Sheet sheet = workbook.getSheetAt(0);

// 读取第一行第一列

Row row = sheet.getRow(0);

Cell cell = row.getCell(0);

// 输出单元内容

System.out.println(cell.getStringCellValue());

// 操作结束,关闭文件

is.close();

}

}

2、07

@Test

public void testRead07() throws Exception{

InputStream is = new FileInputStream("d:/excel-poi/商品表-07.xlsx");

Workbook workbook = new XSSFWorkbook(is);

......

}

3、读取不同类型的数据

1、读取不同的数据类型

@Test

public void testCellType() throws Exception {

InputStream is = new FileInputStream("d:/excel-poi/会员消费商品明细表.xls");

Workbook workbook = new HSSFWorkbook(is);

Sheet sheet = workbook.getSheetAt(0);

// 读取标题所有内容

Row rowTitle = sheet.getRow(0);

if (rowTitle != null) {// 行不为空

// 读取cell

int cellCount = rowTitle.getPhysicalNumberOfCells();

for (int cellNum = 0; cellNum < cellCount; cellNum++) {

Cell cell = rowTitle.getCell(cellNum);

if (cell != null) {

int cellType = cell.getCellType();

String cellValue = cell.getStringCellValue();

System.out.print(cellValue + "|");

}

}

System.out.println();

}

// 读取商品列表数据

int rowCount = sheet.getPhysicalNumberOfRows();

for (int rowNum = 1; rowNum < rowCount; rowNum++) {

Row rowData = sheet.getRow(rowNum);

if (rowData != null) {// 行不为空

// 读取cell

int cellCount = rowTitle.getPhysicalNumberOfCells();

for (int cellNum = 0; cellNum < cellCount; cellNum++) {

System.out.print("【" + (rowNum + 1) + "-" + (cellNum + 1) + "】");

Cell cell = rowData.getCell(cellNum);

if (cell != null) {

int cellType = cell.getCellType();

//判断单元格数据类型

String cellValue = "";

switch (cellType) {

case HSSFCell.CELL_TYPE_STRING://字符串

System.out.print("【STRING】");

cellValue = cell.getStringCellValue();

break;

case HSSFCell.CELL_TYPE_BOOLEAN://布尔

System.out.print("【BOOLEAN】");

cellValue = String.valueOf(cell.getBooleanCellValue());

break;

case HSSFCell.CELL_TYPE_BLANK://空

System.out.print("【BLANK】");

break;

case HSSFCell.CELL_TYPE_NUMERIC:

System.out.print("【NUMERIC】");

//cellValue = String.valueOf(cell.getNumericCellValue());

if (HSSFDateUtil.isCellDateFormatted(cell)) {//日期

System.out.print("【日期】");

Date date = cell.getDateCellValue();

cellValue = new DateTime(date).toString("yyyy-MM-dd");

} else {

// 不是日期格式,则防止当数字过长时以科学计数法显示

System.out.print("【转换成字符串】");

cell.setCellType(HSSFCell.CELL_TYPE_STRING);

cellValue = cell.toString();

}

break;

case Cell.CELL_TYPE_ERROR:

System.out.print("【数据类型错误】");

break;

}

System.out.println(cellValue);

}

}

}

}

is.close();

}

2、计算公式

@Test

public void testFormula() throws Exception{

InputStream is = new FileInputStream("d:/excel-poi/计算公式.xls");

Workbook workbook = new HSSFWorkbook(is);

Sheet sheet = workbook.getSheetAt(0);

// 读取第五行第一列

Row row = sheet.getRow(4);

Cell cell = row.getCell(0);

//公式计算器

FormulaEvaluator formulaEvaluator = new HSSFFormulaEvaluator((HSSFWorkbook) workbook);

// 输出单元内容

int cellType = cell.getCellType();

switch (cellType) {

case Cell.CELL_TYPE_FORMULA://2

//得到公式

String formula = cell.getCellFormula();

System.out.print(formula);

CellValue evaluate = formulaEvaluator.evaluate(cell);

//String cellValue = String.valueOf(evaluate.getNumberValue());

String cellValue = evaluate.formatAsString();

System.out.println(cellValue);

break;

}

}

4.EasyExcel使用

4.1、Excel导入导出的应用场景

1、数据导入:减轻录入工作量

2、数据导出:统计信息归档

3、数据传输:异构系统之间数据传输

4.2、EasyExcel简介

Java领域解析、生成Excel比较有名的框架有Apache poi、jxl等。但他们都存在一个严重的问题就是非常的耗内存。如果你的系统并发量不大的话可能还行,但是一旦并发上来后一定会OOM或者JVM频繁的full gc。

EasyExcel是阿里巴巴开源的一个excel处理框架,以使用简单、节省内存著称。EasyExcel能大大减少占用内存的主要原因是在解析Excel时没有将文件数据一次性全部加载到内存中,而是从磁盘上一行行读取数据,逐个解析。

EasyExcel采用一行一行的解析模式,并将一行的解析结果以观察者的模式通知处理(AnalysisEventListener)。

5.EasyExcel写入

5.1 创建项目,实现EasyExcel对Excel写操作

1、创建一个普通的maven项目

项目名:excel-easydemo

2、pom中引入xml相关依赖

com.alibaba

easyexcel

2.1.1

3、创建实体类

设置表头和添加的数据字段

import com.alibaba.excel.annotation.ExcelProperty;

//设置表头和添加的数据字段

public class DemoData {

//设置表头名称

@ExcelProperty("学生编号")

private int sno;

//设置表头名称

@ExcelProperty("学生姓名")

private String sname;

public int getSno() {

return sno;

}

public void setSno(int sno) {

this.sno = sno;

}

public String getSname() {

return sname;

}

public void setSname(String sname) {

this.sname = sname;

}

@Override

public String toString() {

return "DemoData{" +

"sno=" + sno +

", sname='" + sname + '\'' +

'}';

}

}

4、实现写操作

(1)创建方法循环设置要添加到Excel的数据

//循环设置要添加的数据,最终封装到list集合中

private static List data() {

List list = new ArrayList();

for (int i = 0; i < 10; i++) {

DemoData data = new DemoData();

data.setSno(i);

data.setSname("张三"+i);

list.add(data);

}

return list;

}

(2)实现最终的添加操作(写法一)

public static void main(String[] args) throws Exception {

// 写法1

String fileName = "F:\\11.xlsx";

// 这里 需要指定写用哪个class去写,然后写到第一个sheet,名字为模板 然后文件流会自动关闭

// 如果这里想使用03 则 传入excelType参数即可

EasyExcel.write(fileName, DemoData.class).sheet("写入方法一").doWrite(data());

}

(3)实现最终的添加操作(写法二)

public static void main(String[] args) throws Exception {

// 写法2,方法二需要手动关闭流

String fileName = "F:\\112.xlsx";

// 这里 需要指定写用哪个class去写

ExcelWriter excelWriter = EasyExcel.write(fileName, DemoData.class).build();

WriteSheet writeSheet = EasyExcel.writerSheet("写入方法二").build();

excelWriter.write(data(), writeSheet);

/// 千万别忘记finish 会帮忙关闭流

excelWriter.finish();

}

6.EasyExcel读取

6.1、实现EasyExcel对Excel读操作

1、创建实体类

import com.alibaba.excel.annotation.ExcelProperty;

public class ReadData {

//设置列对应的属性

@ExcelProperty(index = 0)

private int sid;

//设置列对应的属性

@ExcelProperty(index = 1)

private String sname;

public int getSid() {

return sid;

}

public void setSid(int sid) {

this.sid = sid;

}

public String getSname() {

return sname;

}

public void setSname(String sname) {

this.sname = sname;

}

@Override

public String toString() {

return "ReadData{" +

"sid=" + sid +

", sname='" + sname + '\'' +

'}';

}

}

2、创建读取操作的监听器

import com.alibaba.excel.context.AnalysisContext;

import com.alibaba.excel.event.AnalysisEventListener;

import com.alibaba.excel.exception.ExcelDataConvertException;

import com.sun.scenario.effect.impl.sw.sse.SSEBlend_SRC_OUTPeer;

import java.util.ArrayList;

import java.util.List;

import java.util.Map;

//创建读取excel监听器

public class ExcelListener extends AnalysisEventListener {

//创建list集合封装最终的数据

List list = new ArrayList();

//一行一行去读取excle内容

@Override

public void invoke(ReadData user, AnalysisContext analysisContext) {

System.out.println("***"+user);

list.add(user);

}

//读取excel表头信息

@Override

public void invokeHeadMap(Map headMap, AnalysisContext context) {

System.out.println("表头信息:"+headMap);

}

//读取完成后执行

@Override

public void doAfterAllAnalysed(AnalysisContext analysisContext) {

}

}

3、调用实现最终的读取

public static void main(String[] args) throws Exception {

// 写法1:

String fileName = "F:\\01.xlsx";

// 这里 需要指定读用哪个class去读,然后读取第一个sheet 文件流会自动关闭

EasyExcel.read(fileName, ReadData.class, new ExcelListener()).sheet().doRead();

// 写法2:

InputStream in = new BufferedInputStream(new FileInputStream("F:\\01.xlsx"));

ExcelReader excelReader = EasyExcel.read(in, ReadData.class, new ExcelListener()).build();

ReadSheet readSheet = EasyExcel.readSheet(0).build();

excelReader.read(readSheet);

// 这里千万别忘记关闭,读的时候会创建临时文件,到时磁盘会崩的

excelReader.finish();

}

你可能感兴趣的:(easyexcel和poi对比)