导入

准备工作

见:https://www.cnblogs.com/wywblogs/p/16095576.html

异步导入

接口代码

public Map importMemberList(@RequestPart("file") MultipartFile file) {
log4.info("数据导入");
ImportTask importTask = new ImportTask();
importTask.setCreateBy(getCurrentUser.get().getID());
// 读取每一行的监听器
ExcelDataListener<AreaImportDto> excelDataListener = new ExcelDataListener<>();
// excel文件校验
Map<String, Object> check = ExcelUtil.importCheck(file, importTask, excelDataListener, AreaImportDto.class, "导入功能模块名");
// 保存导入记录
importTaskMapper.save(importTask);
// 提交事务,否则下面异步service查询不到保存的importTask
sqlSessionTemplate.flushStatements();
String taskCode = importTask.getCode();
// 如果校验不通过则不进行导入
if (!check.get("code").equals(200)) {
check.put("taskCode",taskCode);
return check;
}
// 进行导入
iAreaService.importMemberList(getCurrentUser.get().getID(), excelDataListener, taskCode);
return new HashMap<String, Object>() {{
put("msg", "数据导入成功!详情请查看导入任务列表");
put("taskCode", taskCode);
put("code", 200);
}};
}

ps:

AreaImportDto:导入的实体类

ImportTask :保存导入任务记录的实体类(包含错误信息)

导入工具类方法

/**
* 校验导入的数据正确性
* @param file 上传的文件
* @param importTask 导入任务记录
* @param excelDataListener 导入监听器
* @param clazz 导入的类对象
* @param importType 导入类型(属于哪个模块的导入)
* @param <T> 导入实体类类型
* @return
*/
public static <T> Map<String, Object> importCheck(MultipartFile file,ImportTask importTask, ExcelDataListener<T> excelDataListener,
Class<T> clazz, String importType) {
SimpleDateFormat sdf = new SimpleDateFormat("yyyyMMddHHmmss");
Date date = new Date();
importTask.setImportType(importType);
importTask.setCode(sdf.format(date));
importTask.setImportTime(date);
importTask.setFileName(file.getOriginalFilename());
importTask.setCreateAt(date);
importTask.setImportStart(date);
HashMap<String, Object> result = new HashMap<String, Object>() {{
put("code", 200);
}};
try {
if (!checkExcelType(file)) {
importTask.setErrorMessage(importType+"数据导入文件类型错误");
importTask.setStatus(2l);
result.put("code", 401);
result.put("msg", importType+"数据导入文件类型错误");
} else {
// 异步执行上传的file临时文件会被清理掉,在异步service中会获取不到文件
EasyExcel.read(file.getInputStream(),excelDataListener)
.head(clazz)
.sheet()
.doRead();
List<String> errorCol = checkExcelHead(clazz, excelDataListener.getHeadMap());
if (errorCol.size() > 0) {
importTask.setErrorMessage(importType+"数据导入文件列名错误:" + String.join(",",errorCol));
importTask.setStatus(2l);
result.put("code", 401);
result.put("msg", importType+"数据导入文件列名错误");
} else {
// 导入的数据的正确性
List<String> nullFields = excelDataListener.getNullFields();
if (nullFields.size() > 0) {
importTask.setErrorMessage(importType+"数据导入文件必要列为空:"+
String.join(";",nullFields) + ";");
importTask.setStatus(2l);
if (excelDataListener.getDataList().size() == 0) {
result.put("code", 401);
result.put("msg", importType+"数据导入文件必要列为空:"+
String.join(";",nullFields));
}
}
}
}
} catch (Exception e) {
e.printStackTrace();
log4.error(importType+"数据导入文件读取异常"+e.getMessage());
importTask.setErrorMessage(importType+"数据导入文件读取异常"+e.getMessage());
importTask.setStatus(2l);
result.put("code", 401);
result.put("msg", importType+"数据导入文件读取异常"+e.getMessage());
}
return result;
} /**
* excel文件类型校验
*
* @param file
* @return
*/
public static boolean checkExcelType(MultipartFile file) {
String filename = file.getOriginalFilename();
if (filename == null)
return false;
return filename.endsWith("xlsx") || filename.endsWith("xls");
} /**
* excel表头校验
* @param clazz 导入实体的类对象
* @param headMap 导入excel的表头
* @return
*/
public static List<String> checkExcelHead(Class<?> clazz,Map<Integer, String> headMap) {
Field[] fields = clazz.getDeclaredFields();
ArrayList<String> errorCol = new ArrayList<>();
// 实体类的head
List<String> entityHead = new ArrayList<>();
for (Field field : fields) {
ExcelProperty annotation = field.getDeclaredAnnotation(ExcelProperty.class);
if (annotation != null) {
List<String> head = Arrays.asList(annotation.value());
entityHead.addAll(head);
}
}
for (Map.Entry<Integer, String> entry : headMap.entrySet()) {
// 解析的表头是否在实体类中
if (!entityHead.contains(entry.getValue()))
errorCol.add(entry.getValue());
}
return errorCol;
}

service开始进行数据导入

/**
* excel数据导入
* @param userId 当前用户id (由于异步方法执行时注入的用户信息service获取不到,因此要在controller中获取后传入)
* @param excelDataListener excel读取监听器
* @param taskCode 导入任务记录的code
*/
@Async
@Transactional(rollbackFor = Exception.class)
public void importMemberList(String userId,ExcelDataListener<AreaImportDto> excelDataListener,String taskCode) {
// 导入任务记录
ImportTask importTask = importTaskMapper.findByCode(taskCode);
log4.info("开始进行数据导入");
// 解析错误的数据
List<Map<String, String>> errorDataList = excelDataListener.getErrorDataList();
if (!CollectionUtils.isEmpty(errorDataList)) {
importTask.setErrorData(errorDataList.toString() + ";");
importTask.setErrorMessage(String.join(";",excelDataListener.getErrorMsg())+ ";");
importTask.setStatus(2l);
} else {
importTask.setStatus(1l);
} // 解析成功的数据
List<AreaImportDto> dataList = excelDataListener.getDataList(); try {
// 将导入实体类转为数据表对应的实体类
// 调用service方法进行保存(其中出错或者出现异常的数据根据起行索引字段构造错误信息保存在importTask中)
} catch (Exception e) {
String errorMessage = importTask.getErrorMessage();
importTask.setErrorMessage(errorMessage + ";数据导入异常");
importTask.setStatus(2l);
}
importTask.setImportEnd(new Date());
importTask.setDataNum((long)(dataList.size()) + (long)(errorDataList.size()));
// 保存任务记录
importTaskMapper.update(importTask);
log4.info("数据导入完成!");
}

EasyExcel实现文件导入的更多相关文章

  1. 按照TYPE的文件导入导出功能

    /** * 导入文件Action;*/private File excelFile;// 保存原始文件名private String excelFileFileName;// 保存原始文件名priva ...

  2. 将.dat文件导入数据库

    *最近在搞文本分类,就是把一批文章分成[军事].[娱乐].[政治]等等. 但是这个先需要一些样本进行训练,感觉文本分类和"按图索骥"差不多,训练的文章样本就是"图&quo ...

  3. 怎样将多个CSS文件导入一个CSS文件中

    问题: 在HTML中引入css的其中的两个方法:    导入式和链接式的目的都是将一个独立的css文件引入一个文件中,二者的区别不大,事实上,二者最大的区别在于链接式使用html的标记引入外部css文 ...

  4. 将DBF文件导入Sqlserver数据库

    项目中的问题:用户选择N个dbf文件导入sql2005数据库,由于每年dbf表结构都在变化,所以在sql2005中根本就不存在,需要每年根据dbf的结构自动建表.(文章来自http://blog.cs ...

  5. [转载]将别人的项目或JAVA文件导入到自己的Eclipse中时,常常会出现JAVA文件的中文注释变成乱码的情况,解决办法

    eclipse 代码中文注释乱码 求解决 将别人的项目或JAVA文件导入到自己的Eclipse中时,常常会出现JAVA文件的中文注释变成乱码的情况,主要原因就是别人的IDE编码格式和自己的Eclips ...

  6. unity3d教程游戏包含的一切文件导入资源

    http://www.58player.com/blog-2327-954.html 导入资源 将文件增加至工程文件夹的资源 (Assets) 文件夹后,Unity 将自动检测文件.将任何资源 (As ...

  7. Apache Spark技术实战之4 -- 利用Spark将json文件导入Cassandra

    欢迎转载,转载请注明出处. 概要 本文简要介绍如何使用spark-cassandra-connector将json文件导入到cassandra数据库,这是一个使用spark的综合性示例. 前提条件 假 ...

  8. 【转】SVN的dump文件导入

    转载地址:http://erniu.sz.blog.163.com/blog/static/11517292220103282813176/ 把SVN的dump文件导入SVN数据库的方法: 在SVN ...

  9. python 从文件导入分类

    # -*- coding:utf-8 -*- """ 从文件导入分类 根据行首制表符或空格确定层级关系(4个空格等于一个制表符 同一行制表符和空格不能混用 ) 必须是 u ...

随机推荐

  1. CF1612D X-Magic Pair

    题意: 给一个数对 \((a,b)\) ,每次可以进行操作 \((a,b) \to (|a-b|,b)\) 或 \((a,b) \to (a,∣a−b∣)\),问最后能否令 \(a=x\) 或 \(b ...

  2. WPF 实现用户头像选择器

    制作一个用户头像选择器仿 WeGame 制作一个用户头像选择Canvas为父控件所实现,展示图片使用Image,Path当作上方的蒙版; Canvas:主要用途方便移动Image,设置ClipToBo ...

  3. srec_cat 常用参数的使用

    前言 下面介绍映像文件工具 srec_cat 的使用,如何通过相关参数实现自己需要的功能. 下载链接:SRecord 使用方式 文件类型 在输入文件和输出文件文件时要指明文件类型,常用的如: test ...

  4. 第十七天python3 文件IO(三)

    CSV文件 csv是一个被行分隔符.列分隔符化分成行和列的文本文件: csv不指定字符编码: 行分隔符为\r\n,最后一行可以没有换行符: 列分隔符常为逗号或者制表符: 每一行称为一条记录record ...

  5. list集合的介绍和常用方法

    List接口介绍 java.util.List接口继承自Collection接口,是单列集合的一个重要分支,习惯性地会将实现了List接口的对象成为List集合.在List集合中允许出现重复的元素,所 ...

  6. windows下memcache安装

    Windows下的Memcache安装:1. 下载memcache的windows稳定版,解压放某个盘下面,比如在c:memcached2. 在终端(也即cmd命令界面)下输入 'c:memcache ...

  7. 选择结构——嵌套 if 控制语句

    1.嵌套 if 控制语句 概念: 在 if 控制语句中又包含一个或多个 if 控制语句的简称为嵌套 if 控制语句.嵌套 if 控制语句可以通过外层语句和内层语句的协作,来增强程序的灵活性. 语法格式 ...

  8. Computational Protein Design with Deep Learning Neural Networks

    本文使用深度神经网络完成计算蛋白质设计去预测20种氨基酸概率. Introduction 针对特定结构和功能的蛋白质进行工程和设计,不仅加深了对蛋白质序列结构关系的理解,而且在化学.生物学和医学等领域 ...

  9. 先导,对IOC容器的理解

    先导,对IOC容器的理解 通俗的讲就是把你的class类交给spring的IOC容器去管理 需要对该类的属性注入一些值,就可以通过spring提供的xml文件或者注解进行注入 自己使用时在IOC容器工 ...

  10. Gumbel_Softmax 概要

    Gumble_Softmax 可以解决的问题 场景:对于一个分类任务,通常会使用softmax函数来将模型的输出值转换为概率的形式,并通过argmax函数取最大的概率值标签作为模型的预测标签.在分类任 ...