以ORM的思路来从Excel文件中读取JSON数据列表
1.一个常见的问题就是如何读取excel。
这里面有几个分支的问题,一个是如何使用poi读取excel,网上例子很多,但是这只解决了第一步。如何将excel读取入一定的数据结构这是第二个问题,还有就是读取到数据了以后如何存放,如何将其转换成我们希望的对象?这些问题一个比一个在应用的更高层级上,也就是说,如果excel是数据源,如何将数据源读取到的数据转换成我们喜爱的实体类,也就是ORM的思想,只是在读取excel这个事情上,我们的数据源是个文件,需要一定的驱动去读取,仅此而已。
2..破解问题
我们需要poi包的支持,这是世界java圈里公认的组件,不用再次发明轮子。我们需要用到json包,用alibaba出品的fastjson包,这个也非常好用。感谢zj的推荐。思考的过程是这样的,在我们现有的知识里面,已经有这样几个概念了,excel通过poi可以读取它的值,json包可以把string直接转换成pojo,那么我们将他们建立联系,从excel中读取行到json,再把json变成实体类,就可以达到我们预想的效果了。那么还有一个问题需要解决,就是如何指定类的属性与excel对应列的关系。仔细想了一阵子,还没想到好的解决办法,如果没有办法自动,那就可以用指定的方法,外部指定其实已经解耦了,只是没那么完美而已。
3.解决问题
第一步,我们建立pojo类属性与excel的对照关系,我们还是利用json作为中间传递的载体。
/**
* 创建用于翻译中文列名到英文变量名的对照字典。
* @param keys 中文或默认的excel列名称
* @param values 变量名
* @return
* @throws Exception
*/
public JSONObject genImportKyes(String[] keys,String[] values) throws Exception {
try {
JSONObject result = new JSONObject();
for (int i = 0; i < keys.length; i++) {
result.put(keys[i], values[i]);
}
return result;
}catch (Exception e){
e.printStackTrace();
throw e;
}
}
调用的时候比如excel有字段"序号", "数量", "日期", "小计", "备注",而我们的pojo对象的属性是"id","count","date","sum","memo",我们这样调用
JSONObject keys = eo.genImportKyes(new String[]{"序号", "数量", "日期", "小计", "备注"},new String[]{"id","count","date","sum","memo"} );
这样这个keys里面我们就建立了excel中的列名和我们实体类属性的对应关系了。
第二步,从excel的指定行去读取一行,来建立属性名和excel列的位置的对应关系。也就是我们需要知道我们的序号兑换成的id这个数据是放在excel的第几列里面的。
public JSONObject createRowIndex(Row row, JSONObject keys) {
int col_count = row.getLastCellNum();
JSONObject jsonObject = new JSONObject();
for (int i = 0; i < col_count; i++) {
String cv = row.getCell(i).getStringCellValue();
if (keys == null) {
jsonObject.put(cv, i);
} else {
String key = keys.getString(cv);
jsonObject.put(key, i);
}
}
return jsonObject;
}
通过读取excel的一个row放进这个函数,再加上我们第一步准备好的keys,我们就可以比较容易的做出对应关系来。因为第一步我们解决了excel列明与实体类属性的对照关系,在第二步我们解决了属性与excel的列的对应位置关系,接下来我们就可以去读取格子了,因为我们需要的信息都准备好了。也就是我们已经建立好了读取的规则。
第三步:读取excel的其他所有行,并按照第一步建立的关系创建出json对象来。这中间有一个需要注意的地方是我们队与excel中的特殊类型需要额外的关照一下,比如日期类型、数字类型等等。具体可以看源码public String parseCell(Cell cell) 这个函数的实现。我们通过每一行的读取来获得一个json的对象,接下来我们在使用json包转javaobject的能力把它变成对应的实体类。接近成功了。
4.最后我们来看看,假设我们有一个excel文件是这样的

我们希望读取出来以后的实体类是这样的。

是时候展示真正的实力啦。
@Test
public void test1() throws Exception {
excelOpterator eo = new excelOpterator().setHeadRowIndex(0);
JSONObject keys = eo.genImportKyes(new String[]{"序号", "数量", "日期", "小计", "备注"},new String[]{"id","count","date","sum","memo"} );
JSONArray rows = eo.readXls("C:/excel.xls", keys);
for(Object jo :rows){
testEntity to = JSON.toJavaObject((JSONObject)jo,testEntity.class );
System.out.println(to);
}
}
看到输出的时候总是很开心,但是回头看自己思考的过程,也会比较有意思,希望大家都能享受到这个轻松的结果。bingo

文末有福利
https://gitee.com/youliaoo/lutraExcelHelper
以ORM的思路来从Excel文件中读取JSON数据列表的更多相关文章
- 记新人从 excel 文件中读取字典数据踩的一个坑
原本是打算今天分享一下最近学习接口自动化的心得体会,然而在我写模板的时候,却被一个坑拦我大半天,心得体会不得不 延期再分享了.在我无数次调试无数次看log,终于发现并解决这个问题了.下面记录一下踩的坑 ...
- 条形码的应用三-----------从Excel文件中读取条形码
条形码的应用三------从Excel文件中读取条形码 介绍 上一篇文章,我向大家展示了生成多个条形码并存储到Excel文件中的一个方法.后来我又有了个想法:既然条码插入到excel中了,我可不可以从 ...
- 从Excel文件中读取内容
从Excel文件中读取内容 global::System.Web.HttpPostedFileBase file = Request.Files["txtFile"]; strin ...
- go从文件中读取json字符串并转换
go从文件中读取json字符串并转换 将要读取的文件的一部分 [ { "children": [ { "children": [ { "code&qu ...
- vue-cli项目 build后请求本地static文件中的 json数据,路径不对,报错404处理方法
vue-cli 项目 build 出错点: 1,build生成dist 放在tomcat上 报错,不显示内容 解决办法: config>index.js===>assetsPublic ...
- java读取url中json文件中的json数据
有时候需要远程从其他接口中获取json数据,如果遇到返回的json数据是一个文件而不直接是数据,那么可以通过以下方法进行读取: /** * 从数据接口获取到数据 * @return * @throws ...
- easyui datagrid 加载静态文件中的json数据
本文主要介绍easyui datagrid 怎么加载静态文件里的json数据,开发环境vs2012, 一.json文件所处的位置 二.json文件内容 {"total":28,&q ...
- springMVC从上传的Excel文件中读取数据
示例:导入客户文件(Excle文件) 一.编辑customer.xlsx 二.在spring的xml文件设置上传文件大小 <!-- 上传文件拦截,设置最大上传文件大小 10M=10*1024*1 ...
- 从文件中读取数组数据————Java
自己总结一下Java文件的读取类似数组数据的方法,自己可以快速查看. 一.规整化数据: 对于数组数据是一一对应的情况 ArrayList<String> arrayList = new A ...
随机推荐
- 4.css基础
1 Css概念 CSS 指层叠样式表 (Cascading Style Sheets)(级联样式表) Css是用来美化html标签的,相当于页面化妆. ◆样式表书写位置 2选择器 2.1 写法 选择器 ...
- Ubuntu16.04安装NVIDIA显卡驱动
1.下载官方驱动程序 http://www.geforce.cn/drivers 如果我们直接安装驱动的话,往往会报错:ERROR: The Nouveau kernel driver is curr ...
- [转载]持续交付和DevOps的前世今生
作者/分享人:乔梁,20年IT老兵,腾讯公司高级管理顾问,敏捷和精益开发专家,持续交付领域先行者.曾就职于百度,国内多个知名互联网公司的企业教练. 历年QCon技术大会的讲师和专题出品人. 这是一个新 ...
- javascript变量浅析
变量声明 javascript 使用var + 变量名 声明变量,因为javascript是弱类型语言, 所有我们可以随意更改已有变量的类型. var b=1; b='2', 另外不同于c#中的var ...
- DEV通过FindFilterText自动检索gridview内容
private void TreeView1_NodeMouseClick(object sender, TreeNodeMouseClickEventArgs e) { if (names!=nul ...
- eclipse代码提示javadoc背景为黑色框的解决办法
我的eclipse是近期下载的oxygen版本.不知道怎么出现了一个这个问题,鼠标悬停指向代码时应该出现的代码提示解释框,全为黑色,看不到文字.如下图 经过验证,最终解决方法为window->G ...
- “全栈2019”Java多线程第三十一章:中断正在等待显式锁的线程
难度 初级 学习时间 10分钟 适合人群 零基础 开发语言 Java 开发环境 JDK v11 IntelliJ IDEA v2018.3 文章原文链接 "全栈2019"Java多 ...
- VMware中安装Contos
1 检查BIOS虚拟化支持 2 新建虚拟机 3 新建虚拟机向导 4 创建虚拟空白光盘 5 安装Linux系统对应的CentOS版 6 虚拟机命名和定位磁盘位置 7 处理器配置,看自己是否是双核.多核 ...
- 通过DHCP动态管理IP地址
DHCP(Dynamic Host Configuration Protocol,动态主机配置协议)是一个局域网的网络协议,使用UDP协议工作, 主要有两个用途:给内部网络或网络服务供应商自动分配IP ...
- spring cloud学习(六) 配置中心-自动更新
上一篇学习了spring cloud config的基本使用,但发现有个问题,就是每次更改配置后,都需要重启服务才能更新配置,这样肯定是不行的.在上网查资料了解后,spring cloud支持通过AM ...