mirror of https://github.com/alibaba/easyexcel
41 changed files with 1201 additions and 221 deletions
@ -0,0 +1,9 @@ |
|||||||
|
# 建议先去看文档 |
||||||
|
[快速开始](https://easyexcel.opensource.alibaba.com/docs/current/) 、[常见问题](https://easyexcel.opensource.alibaba.com/qa/) |
||||||
|
# 异常代码 |
||||||
|
```java |
||||||
|
这里写你的代码 |
||||||
|
``` |
||||||
|
# 异常提示 |
||||||
|
大家尽量把问题一次性描述清楚,然后贴上全部异常,这样方便把问题一次性解决掉。 |
||||||
|
# 其他描述 |
@ -0,0 +1,26 @@ |
|||||||
|
# 通过 Github action, 在仓库的每一次 commit 后自动同步到 Gitee 上 |
||||||
|
name: Mirror the Github organization repos to Gitee |
||||||
|
on: [push, pull_request] |
||||||
|
|
||||||
|
jobs: |
||||||
|
repo-sync: |
||||||
|
runs-on: ubuntu-latest |
||||||
|
steps: |
||||||
|
- uses: actions/checkout@main |
||||||
|
with: |
||||||
|
persist-credentials: false |
||||||
|
- name: Mirror the Github organization repos to Gitee. |
||||||
|
uses: Yikun/hub-mirror-action@master |
||||||
|
with: |
||||||
|
# 必选,需要同步的 Github 这里记住选择的是仓库 或者账号 而不是具体的项目 |
||||||
|
src: github/alibaba |
||||||
|
# 必选,需要同步到的 Gitee 这里记住选择的是仓库 或者账号 而不是具体的项目 |
||||||
|
dst: gitee/easyexcel |
||||||
|
# 必选,Gitee公钥对应的私钥,https://gitee.com/profile/sshkeys |
||||||
|
dst_key: ${{ secrets.GITEE_PRIVATE_KEY }} |
||||||
|
# 必选,Gitee对应的用于创建仓库的token,https://gitee.com/profile/personal_access_tokens |
||||||
|
dst_token: ${{ secrets.GITEE_TOKEN }} |
||||||
|
# 如果是组织,指定组织即可,默认为用户 user |
||||||
|
account_type: org |
||||||
|
# 需要同步的仓库里面的项目 |
||||||
|
static_list: "easyexcel" |
@ -0,0 +1,19 @@ |
|||||||
|
package com.alibaba.excel.constant; |
||||||
|
|
||||||
|
import java.math.MathContext; |
||||||
|
import java.math.RoundingMode; |
||||||
|
|
||||||
|
/** |
||||||
|
* Used to store constant |
||||||
|
* |
||||||
|
* @author Jiaju Zhuang |
||||||
|
*/ |
||||||
|
public class EasyExcelConstants { |
||||||
|
|
||||||
|
/** |
||||||
|
* Excel by default with 15 to store Numbers, and the double in Java can use to store number 17, led to the accuracy |
||||||
|
* will be a problem. So you need to set up 15 to deal with precision |
||||||
|
*/ |
||||||
|
public static final MathContext EXCEL_MATH_CONTEXT = new MathContext(15, RoundingMode.HALF_UP); |
||||||
|
|
||||||
|
} |
@ -0,0 +1,40 @@ |
|||||||
|
package com.alibaba.excel.enums; |
||||||
|
|
||||||
|
import java.math.BigDecimal; |
||||||
|
import java.time.LocalDateTime; |
||||||
|
import java.util.HashMap; |
||||||
|
import java.util.Map; |
||||||
|
|
||||||
|
import com.alibaba.excel.metadata.data.CellData; |
||||||
|
import com.alibaba.excel.util.StringUtils; |
||||||
|
|
||||||
|
/** |
||||||
|
* Read not to {@code com.alibaba.excel.metadata.BasicParameter#clazz} value, the default will return type. |
||||||
|
* |
||||||
|
* @author Jiaju Zhuang |
||||||
|
*/ |
||||||
|
public enum ReadDefaultReturnEnum { |
||||||
|
/** |
||||||
|
* default.The content of cells into string, is the same as you see in the excel. |
||||||
|
*/ |
||||||
|
STRING, |
||||||
|
|
||||||
|
/** |
||||||
|
* Returns the actual type. |
||||||
|
* Will be automatically selected according to the cell contents what return type, will return the following class: |
||||||
|
* <ol> |
||||||
|
* <li>{@link BigDecimal}</li> |
||||||
|
* <li>{@link Boolean}</li> |
||||||
|
* <li>{@link String}</li> |
||||||
|
* <li>{@link LocalDateTime}</li> |
||||||
|
* </ol> |
||||||
|
*/ |
||||||
|
ACTUAL_DATA, |
||||||
|
|
||||||
|
/** |
||||||
|
* Return to {@link com.alibaba.excel.metadata.data.ReadCellData}, can decide which field you need. |
||||||
|
*/ |
||||||
|
READ_CELL_DATA, |
||||||
|
; |
||||||
|
|
||||||
|
} |
@ -0,0 +1,143 @@ |
|||||||
|
package com.alibaba.easyexcel.test.demo.rare; |
||||||
|
|
||||||
|
import java.io.File; |
||||||
|
import java.util.Date; |
||||||
|
import java.util.List; |
||||||
|
|
||||||
|
import com.alibaba.easyexcel.test.demo.write.DemoData; |
||||||
|
import com.alibaba.easyexcel.test.util.TestFileUtil; |
||||||
|
import com.alibaba.excel.EasyExcel; |
||||||
|
import com.alibaba.excel.ExcelWriter; |
||||||
|
import com.alibaba.excel.util.FileUtils; |
||||||
|
import com.alibaba.excel.util.ListUtils; |
||||||
|
import com.alibaba.excel.write.handler.RowWriteHandler; |
||||||
|
import com.alibaba.excel.write.handler.SheetWriteHandler; |
||||||
|
import com.alibaba.excel.write.handler.WorkbookWriteHandler; |
||||||
|
import com.alibaba.excel.write.handler.context.RowWriteHandlerContext; |
||||||
|
import com.alibaba.excel.write.handler.context.SheetWriteHandlerContext; |
||||||
|
import com.alibaba.excel.write.handler.context.WorkbookWriteHandlerContext; |
||||||
|
import com.alibaba.excel.write.metadata.WriteSheet; |
||||||
|
|
||||||
|
import lombok.extern.slf4j.Slf4j; |
||||||
|
import org.apache.poi.ss.usermodel.Cell; |
||||||
|
import org.apache.poi.ss.usermodel.Row; |
||||||
|
import org.apache.poi.ss.usermodel.Sheet; |
||||||
|
import org.apache.poi.ss.usermodel.Workbook; |
||||||
|
import org.apache.poi.xssf.streaming.SXSSFWorkbook; |
||||||
|
import org.junit.Ignore; |
||||||
|
import org.junit.Test; |
||||||
|
|
||||||
|
/** |
||||||
|
* 记录一些不太常见的案例 |
||||||
|
* |
||||||
|
* @author Jiaju Zhuang |
||||||
|
*/ |
||||||
|
@Ignore |
||||||
|
@Slf4j |
||||||
|
public class WriteTest { |
||||||
|
|
||||||
|
/** |
||||||
|
* 压缩临时文件 |
||||||
|
* 在导出Excel且格式为xlsx的时候会生成一个临时的xml文件,会比较大,再磁盘不太够的情况下,可以压缩。 |
||||||
|
* 当然压缩式耗费性能的 |
||||||
|
*/ |
||||||
|
@Test |
||||||
|
public void compressedTemporaryFile() { |
||||||
|
log.info("临时的xml存储在:{}", FileUtils.getPoiFilesPath()); |
||||||
|
File file = TestFileUtil.createNewFile("rare/compressedTemporaryFile" + System.currentTimeMillis() |
||||||
|
+ ".xlsx"); |
||||||
|
|
||||||
|
// 这里 需要指定写用哪个class去写
|
||||||
|
try (ExcelWriter excelWriter = EasyExcel.write(file, DemoData.class).registerWriteHandler( |
||||||
|
new WorkbookWriteHandler() { |
||||||
|
|
||||||
|
/** |
||||||
|
* 拦截Workbook创建完成事件 |
||||||
|
* @param context |
||||||
|
*/ |
||||||
|
@Override |
||||||
|
public void afterWorkbookCreate(WorkbookWriteHandlerContext context) { |
||||||
|
// 获取到Workbook对象
|
||||||
|
Workbook workbook = context.getWriteWorkbookHolder().getWorkbook(); |
||||||
|
// 只有SXSSFWorkbook模式才会生成临时文件
|
||||||
|
if (workbook instanceof SXSSFWorkbook) { |
||||||
|
SXSSFWorkbook sxssfWorkbook = (SXSSFWorkbook)workbook; |
||||||
|
// 设置临时文件压缩,当然这个会浪费cpu性能 但是临时文件会变小
|
||||||
|
sxssfWorkbook.setCompressTempFiles(true); |
||||||
|
} |
||||||
|
} |
||||||
|
}).build()) { |
||||||
|
// 这里注意 如果同一个sheet只要创建一次
|
||||||
|
WriteSheet writeSheet = EasyExcel.writerSheet("模板").build(); |
||||||
|
// 10万数据 确保有足够的空间
|
||||||
|
for (int i = 0; i < 10000; i++) { |
||||||
|
// 分页去数据库查询数据 这里可以去数据库查询每一页的数据
|
||||||
|
List<DemoData> data = data(); |
||||||
|
excelWriter.write(data, writeSheet); |
||||||
|
} |
||||||
|
log.info("写入完毕,开始准备迁移压缩文件。"); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* 在指定单元格写入数据 |
||||||
|
*/ |
||||||
|
@Test |
||||||
|
public void specifiedCellWrite() { |
||||||
|
File file = TestFileUtil.createNewFile("rare/specifiedCellWrite" + System.currentTimeMillis() |
||||||
|
+ ".xlsx"); |
||||||
|
|
||||||
|
// 需要区分是在 最后一行之前 还是之后
|
||||||
|
// 区分的原因是:excel只能一直向前,而且内存里面只存储100条,而afterRowDispose是在每一行写入完成的时候调用,所以修改一行需要拦截这个事件
|
||||||
|
// 如果是在最后一行之后,由于后面不会再有数据了,所以只要拦截afterWorkbookDispose,在整个excel快写完的时候调用,继续写入数据即可
|
||||||
|
|
||||||
|
EasyExcel.write(file, DemoData.class) |
||||||
|
// 写入的值在最后一行之前
|
||||||
|
.registerWriteHandler(new RowWriteHandler() { |
||||||
|
@Override |
||||||
|
public void afterRowDispose(RowWriteHandlerContext context) { |
||||||
|
if (context.getRow().getRowNum() == 2) { |
||||||
|
Cell cell = context.getRow().getCell(2); |
||||||
|
if (cell == null) { |
||||||
|
cell = context.getRow().createCell(2); |
||||||
|
} |
||||||
|
cell.setCellValue("测试的第二行数据呀"); |
||||||
|
} |
||||||
|
} |
||||||
|
}) |
||||||
|
// 写入的值 在最后一一行之后
|
||||||
|
.registerWriteHandler(new WorkbookWriteHandler() { |
||||||
|
@Override |
||||||
|
public void afterWorkbookDispose(WorkbookWriteHandlerContext context) { |
||||||
|
Workbook workbook = context.getWriteWorkbookHolder().getWorkbook(); |
||||||
|
Sheet sheet = workbook.getSheetAt(0); |
||||||
|
Row row = sheet.getRow(99); |
||||||
|
if (row == null) { |
||||||
|
row = sheet.createRow(99); |
||||||
|
} |
||||||
|
Cell cell = row.getCell(2); |
||||||
|
if (cell == null) { |
||||||
|
cell = row.createCell(2); |
||||||
|
} |
||||||
|
cell.setCellValue("测试地99行数据呀"); |
||||||
|
} |
||||||
|
}) |
||||||
|
.sheet("模板") |
||||||
|
.doWrite(data()); |
||||||
|
|
||||||
|
log.info("写入到文件完成:{}", file); |
||||||
|
} |
||||||
|
|
||||||
|
private List<DemoData> data() { |
||||||
|
List<DemoData> list = ListUtils.newArrayList(); |
||||||
|
for (int i = 0; i < 10; i++) { |
||||||
|
DemoData data = new DemoData(); |
||||||
|
data.setString("字符串" + i); |
||||||
|
data.setDate(new Date()); |
||||||
|
data.setDoubleData(0.56); |
||||||
|
list.add(data); |
||||||
|
} |
||||||
|
return list; |
||||||
|
} |
||||||
|
|
||||||
|
} |
@ -0,0 +1,35 @@ |
|||||||
|
package com.alibaba.easyexcel.test.temp; |
||||||
|
|
||||||
|
import java.math.BigDecimal; |
||||||
|
import java.util.Date; |
||||||
|
|
||||||
|
import com.alibaba.excel.annotation.ExcelIgnore; |
||||||
|
import com.alibaba.excel.annotation.ExcelProperty; |
||||||
|
|
||||||
|
import lombok.EqualsAndHashCode; |
||||||
|
import lombok.Getter; |
||||||
|
import lombok.Setter; |
||||||
|
|
||||||
|
/** |
||||||
|
* 基础数据类 |
||||||
|
* |
||||||
|
* @author Jiaju Zhuang |
||||||
|
**/ |
||||||
|
@Getter |
||||||
|
@Setter |
||||||
|
@EqualsAndHashCode |
||||||
|
public class DemoData2 { |
||||||
|
@ExcelProperty("字符串标题") |
||||||
|
private String string; |
||||||
|
@ExcelProperty("日期标题") |
||||||
|
private Date date; |
||||||
|
@ExcelProperty("数字标题") |
||||||
|
private Double doubleData; |
||||||
|
@ExcelProperty("数字标题2") |
||||||
|
private BigDecimal bigDecimal; |
||||||
|
/** |
||||||
|
* 忽略这个字段 |
||||||
|
*/ |
||||||
|
@ExcelIgnore |
||||||
|
private String ignore; |
||||||
|
} |
@ -0,0 +1,24 @@ |
|||||||
|
package com.alibaba.easyexcel.test.temp; |
||||||
|
|
||||||
|
import java.time.LocalDateTime; |
||||||
|
import java.util.Date; |
||||||
|
|
||||||
|
import com.alibaba.excel.annotation.ExcelIgnore; |
||||||
|
import com.alibaba.excel.annotation.ExcelProperty; |
||||||
|
|
||||||
|
import lombok.EqualsAndHashCode; |
||||||
|
import lombok.Getter; |
||||||
|
import lombok.Setter; |
||||||
|
|
||||||
|
/** |
||||||
|
* 基础数据类 |
||||||
|
* |
||||||
|
* @author Jiaju Zhuang |
||||||
|
**/ |
||||||
|
@Getter |
||||||
|
@Setter |
||||||
|
@EqualsAndHashCode |
||||||
|
public class DemoData3 { |
||||||
|
@ExcelProperty("日期时间标题") |
||||||
|
private LocalDateTime localDateTime; |
||||||
|
} |
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Loading…
Reference in new issue