POI读取Excel常见问题

       最近在做一个将excel导入到报表中的功能,使用了POI来实现,发现POI使用有诸多不便之处,先记录下来,以后可能考虑使用Openxml。

       1. 数值类型处理

       通过POI取出的数值默认都是double,即使excel单元格中存的是1,取出来的值也是1.0,这就造成了一些问题,如果数据库字段是int,那么就会wrong data type,所以需要对数值类型处理。

Cell cell = null;// 单元格
Object inputValue = null;// 单元格值
if (!isEmpty(cell) && cell.getCellType() == Cell.CELL_TYPE_NUMERIC) {
	long longVal = Math.round(cell.getNumericCellValue());
	if (Double.parseDouble(longVal + ".0") == doubleVal)
		inputValue = longVal;
	else
		inputValue = doubleVal;
}

       这么处理后,单元格中的小数没有变化,如果是整数,也会取到整数。

       2. 日期类型处理

       很遗憾,POI对单元格日期处理很弱,没有针对的类型,日期类型取出来的也是一个double值,所以同样作为数值类型。

Cell cell = null;// 单元格
Object inputValue = null;// 单元格值
if (!isEmpty(cell) && cell.getCellType() == Cell.CELL_TYPE_NUMERIC) {
    if (DateUtil.isCellDateFormatted(c))// 判断单元格是否属于日期格式
	    inputValue = cell.getDateCellValue();//java.util.Date类型
}

       可以判断得到的Date是日期时间、日期还是时间,可以通过cell.getCellStyle().getDataFormat()来判断,这个返回值没有一个常量值来对应,我本机是excel2013,测试结果是日期时间(yyyy-MM-dd HH:mm:ss) - 22,日期(yyyy-MM-dd) - 14,时间(HH:mm:ss) - 21,年月(yyyy-MM) - 17,时分(HH:mm) - 20,月日(MM-dd) - 58,有了这个,可以根据数据库字段类型,处理之后再入库,相当不方便。

       另外,如果单元格数据格式是自定义的日期格式,那么通过DateUtil.isCellDateFormatted(cell)判断不出来,而且该单元格还是一个数值单元格,返回一个double值,这里比较2。针对这种方式,有两种解决方案,第一种,重写DateUtil.isCellDateFormatted(cell)方法,开源的都有源码;第二种,cell.getCellStyle().getDataFormatString()来判断,这个方法会返回格式字符串,通过这个字符串去匹配,再处理。

       3. 数据有效性

       很奇怪,POI能生成数据有效性(下拉列表),却得不到,或者说我没找到方法去得到,蛋疼。

       附单元格数据类型:

常量

说明

取值

Cell.CELL_TYPE_NUMERIC

数值类型

cell.getNumericCellValue()或cell.getDateCellValue()

Cell.CELL_TYPE_STRING

字符串类型

cell.getStringCellValue()或cell.toString()

Cell.CELL_TYPE_BOOLEAN

布尔类型

cell.getBooleanCellValue()

Cell.CELL_TYPE_FORMULA

表达式类型

cell.getCellFormula()

Cell.CELL_TYPE_ERROR

异常类型不知道何时算异常

cell.getErrorCellValue()

Cell.CELL_TYPE_BLANK

空,不知道何时算空

空就不要取值了吧

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏Fundebug

详解JavaScript之神奇的Object.defineProperty

摘要: JavaScript有个很神奇的Object.defineProperty(),了解一下?

815
来自专栏钟绍威的专栏

怎么实现关闭窗口

思路:在java源文件中查找源代码 首先,我们知道JFrame.setDefaultCloseOperation(int operation)中提供了四种方...

3197
来自专栏salesforce零基础学习

salesforce 零基础学习(十七)Trigger用法

看本篇之前可以相应阅读以下Trigger相关文章: 1.https://developer.salesforce.com/page/Trigger_Framew...

3256
来自专栏后端之路

SpringBoot之条件注解

背景 之前写过关于Spring和Maven的profile的区别 maven profile VS spring profile 我们可以通过上述的profil...

2985
来自专栏web前端

Button按钮--inject与provide

inject 和 provider 是vue中的组合选项,需要一起使用。目的是允许一个祖先组件向其所有子孙后代注入依赖(简单地说就是祖先组件向子孙后代传值的一种...

1091
来自专栏xingoo, 一个梦想做发明家的程序员

Oozie分布式工作流——从理论和实践分析使用节点间的参数传递

Oozie支持Java Action,因此可以自定义很多的功能。本篇就从理论和实践两方面介绍下Java Action的妙用,另外还涉及到oozie中actio...

2755
来自专栏difcareer的技术笔记

彻底弄懂dalvik字节码【一】

之前曾经简单跟踪过代码,知道dalvik的字节码是可以支持解释执行的,所谓的解释执行,其实就是c/c++编写的用于解释并执行dalvik字节码的程序,说白了就是...

1032
来自专栏张善友的专栏

线程安全的Generic Dictionary

System.Collections.Generic.Dictionary<,> 只要不修改该集合,Dictionary 就可以同时支持多个阅读器。即便如此,从...

1996
来自专栏码匠的流水账

聊聊eureka client的fetch-remote-regions-registry属性

本文主要研究一下eureka client的fetch-remote-regions-registry属性

571
来自专栏Java 技术分享

JDBC(MySQL)一周学习总结(二)

37610

扫码关注云+社区