Java解析上传的zip文件--包含Excel解析与图片上传

前言:今天遇到一个需求:上传一个zip格式的压缩文件,该zip中包含人员信息的excel以及excel中每行对应的人的图片,现在需要将该zip压缩包中所有内容解析导入到数据库中,包括图片,并将图片与excel内容对应。

代码演示:

/**
* 信息导入Controller
*/
@RestController
@RequestMapping("/import")
public class ImportController { @AutoWired
private IExcelService excelService /**
* 接收zip
* zip中包含了人员excel以及excel中每行对应的人员图片--默认每个人员图片的名称为number号
*/
@PostMapping(value = "/zip")
@Transactional
public void sendRequest(@RequestParam("file") MultipartFile zipFile) throws IOException {
//通过zip名称创建一个file文件-该文件无具体路径
File file = new File(Objects.requireNonNull(zipFile.getOriginalFilename()));
//将zip写入到file中
FileUtils.writeByteArrayToFile(file, zipFileFile.getBytes());
//设定字符集编码--这一步必须,否则放到linux服务器中会有字符集报错
Charset charset = Charset.forName("GBK");
ZipFile zipFile = new ZipFile(file, charset);
//开始读取zip中文件
Enumeration<? extends ZipEntry> entries = zipFile.entries();
//创建excelZip存放zip中的excel文件内容
List<ZipEntry> excelZip = new ArrayList<ZipEntry>();
//创建imgZip存放zip中的图片文件
List<ZipEntry> imgZip = new ArrayList<ZipEntry>(); //开始读取zip文件
while (entries.hasMoreElements()) {
ZipEntry entry = entries.nextElement();
//通过文件名称去过滤出excel
if (entry.getName().contains("/excel表格名称.xlsx")) {
excelZip.add(entry);
}
//获取包含图片目录下所有的图片信息
if (!entry.isDirectory() && entry.getName().contains("图片目录的名称")) {
imgZip.add(entry);
}
}
//创建文件输入流,将excel变为流
InputStream excelInputStream = zipFile.getInputStream(excelZip.get(0));
//创建EasyExcel中的监听器
ExcelListener excelListener = new ExcelListener();
//调用EasyExcel中的EasyExcelFactory方法去读取输入流
EasyExcelFactory.read(excelInputStream,Excel.class,excelListener).sheet().doRead();
//到此完成获取压缩包中excel的人员信息
List<Excel> excelList = excelListener.getDataList(); //遍历读取图片
imgZip.stream().forEach(zip -> {
try {
//创建压缩图片的输入流
InputStream imgInputStream = zipFile.getInputStream(zip);
//调用流转文件的方法
File img = stream2file(imgInputStream);
for (Excel excel : excelList)
{
//将去除后缀的压缩图片名称与excel中的人员号码字段做比较
if (removeExtension(zip.getName()).contains(excel.getNum())) {
//调用文件上传接口,上传到服务器目录下,并返回文件存储路径
//这边FileUpload.upload为自己写的一个上传接口,此处省略。。。
//如果接受对象为MultipartFile,还需调用fileToMultipartFile转换
String urlPath =FileUpload.upload(fileToMultipartFile(img));
//进行对象的保存
Excel e=new Excel();
//设置存储的图片地址
e.setPath(urlPath);
//调用存储接口存储图片与excel信息
excelService.insert(e);
}
}
} catch (IOException e) {
throw new RuntimeException(e);
}
});
} //设置文件名前缀;必须至少三个字符
static final String PREFIX = "stream2file";
//设置定义文件的后缀扩展名;如果为null,则将使用后缀".tmp"
//这边为图片格式,所以默认给的后缀为jpg格式
static final String SUFFIX = ".jpg"; /**
* 输入流转文件
* 该方法为在内存中创建临时文件,不进行磁盘存储
*/
public static File stream2file(InputStream in) throws IOException {
//创建临时文件
final File tempFile = File.createTempFile(PREFIX, SUFFIX);
tempFile.deleteOnExit();
try (FileOutputStream out = new FileOutputStream(tempFile)) {
IOUtils.copy(in, out);
}
return tempFile;
} /**
* 截取文件名称--去除后缀名(.jpg等)
*/
public static String removeExtension(String fileName) {
int lastDotIndex = fileName.lastIndexOf('.');
if (lastDotIndex == -1) {
// 如果文件名中没有后缀,则返回原文件名
return fileName;
} else {
// 截取从开头到最后一个`.`字符之前的部分
return fileName.substring(0, lastDotIndex);
}
} /**
* file转MultipartFile
*
* @param file file
* @return MultipartFile
*/
public static MultipartFile fileToMultipartFile(File file) {
MultipartFile result = null;
if (null != file) {
try (FileInputStream input = new FileInputStream(file)) {
result = new MockMultipartFile(file.getName().concat("temp"), file.getName(), "text/plain", input);
} catch (IOException e) {
e.printStackTrace();
}
}
return result;
}
}
}

回顾:

博主解答思路为:

  1. 接收压缩文件
  2. 解析压缩文件,并区分excel与图片文件
  3. 将excel与图片进行匹配,将匹配成功的数据存储到数据库中

要点:

  1. 本次读取几乎都是内存读取,将数据读取到内存中,也在磁盘中建立了临时文件
  2. 解析时需要进行多次类型转换
  3. new ZipFile(file, charset)这一步一定要设定字符集编码,并查看window与linux字符集编码的区别,否则会因为字符集编码问题报错而无法运行

Java解析上传的zip文件--包含Excel解析与图片上传的更多相关文章

  1. Java 上传解压zip文件,并且解析文件里面的excel和图片

    需求:上传一个zip文件,zip文件里面包含一个excel和很多图片,需要把excel里面的信息解析出来保存到表中,同时图片也转化成base64保存到数据库表中. PS:为了方便不同水平的开发人员阅读 ...

  2. 解压上传的zip文件流和文件

    /** * 解压上传的zip文件流 * @param stream * @param outputDirectory */ public static String unzip(InputStream ...

  3. Java使用基本JDK操作ZIP文件以及zip文件的加密、解密等功能

    Java使用基本JDK操作ZIP文件 http://blog.csdn.net/zhyh1986/article/details/7723649 Java解压和压缩带密码的zip文件 http://b ...

  4. 历史上的今天mysql数据库包含详情分类以及图片

    历史上的今天mysql数据库包含详情分类以及图片 https://item.taobao.com/item.htm?spm=a2oq0.12575281.0.0.50111debo71iaJ& ...

  5. 用html5文件api实现移动端图片上传&预览效果

    想要用h5在移动端实现图片上传&预览效果,首先要了解html5的文件api相关知识(所有api只列举本功能所需): 1.Blob对象  Blob表示原始二进制数据,Html5的file对象就继 ...

  6. java解压多目录Zip文件(解决中文乱码问题)--转载

    原文地址:http://zhangyongbo.iteye.com/blog/1749439 import java.io.BufferedOutputStream; import java.io.F ...

  7. JAVA解压.Z及.ZIP文件

    <!-- https://mvnrepository.com/artifact/org.apache.commons/commons-compress --> <dependency ...

  8. 包含修改字体,图片上传等功能的文本输入框-Bootstrap

    通过jQuery Bootstrap小插件,框任何一个div转换变成一个富文本编辑框,主要特色: 在Mac和window平台下自动针对常用操作绑定热键 可以拖拽插入图片,支持图片上传(也可以获取移动设 ...

  9. JAVA通过FTP方式向远程服务器或者客户端上传、下载文件,以及删除FTP服务器上的文件

    1.在目标服务器上搭建FTP服务器 搭建方式有多种大家可以自行选择,例如使用Serv-U或者FTPServer.exe:这里我以FTPServer.exe为例搭建:在目标服务器(这里对应的IP是10. ...

  10. 【java】System.out重定向到文件,并重定向会console上

    重定向到文件: System.setOut(new PrintStream(new File("data\\train.1.scale"))); 重定向回console: //把输 ...

随机推荐

  1. C++ Qt开发:如何使用信号与槽

    在Qt中,信号与槽(Signal and Slot)是一种用于对象之间通信的机制.是Qt框架引以为傲的一项机制,它带来了许多优势,使得Qt成为一个强大且灵活的开发框架之一.信号与槽的关联通过QObje ...

  2. [ARC144E]GCD of Path Weights

    Problem Statement You are given a directed graph $G$ with $N$ vertices and $M$ edges. The vertices a ...

  3. [ABC263A] Full House

    Problem Statement We have five cards with integers $A$, $B$, $C$, $D$, and $E$ written on them, one ...

  4. 小傅哥自研插件,为开发提效80%,已经有8.1k安装量!

    作者:小傅哥 博客:https://bugstack.cn 沉淀.分享.成长,让自己和他人都能有所收获! 哈喽,大家好我是技术UP主小傅哥. 如果你担心维护成本和性能考量,不想使用 BeanUtils ...

  5. 在arm架构的银河麒麟系统部署Redis

    以下是在arm架构的银河麒麟系统上部署Redis的详细步骤: 1. 创建文件夹 首先,在合适的位置创建必要的文件夹.在本例中,我们将创建/opt/redis和/usr/src/redis两个文件夹. ...

  6. C++ Qt开发:SqlTableModel映射组件应用

    Qt 是一个跨平台C++图形界面开发库,利用Qt可以快速开发跨平台窗体应用程序,在Qt中我们可以通过拖拽的方式将不同组件放到指定的位置,实现图形化开发极大的方便了开发效率,本章将重点介绍SqlTabl ...

  7. vue3+ts打开echarts的正确方式

    实例项目使用 vite5 + vue3 + ts,项目地址 vite-vue3-charts,预览地址 https://weizwz.com/vite-vue3-charts 准备工作 1. 注册为百 ...

  8. Sliver 二开准备

    cs被杀麻了,最近打算看看一下sliver的源码进行一下二开,这篇是记录遇到的一些问题 编译sliver Windows下 官方说用MingW,但是我自己用他带的make不行, ​​ ‍ 下载make ...

  9. 2.elasticsearch中的mapping

    mapping 顾名思义,代表了映射关系.是文档中字段和数据类型的映射关系 为什么要了解mapping 虽然elasticsearch中已尽有的动态mapping(Dynamic Mapping),而 ...

  10. buuctf 加固题 babypython WriteUp

    原题wp参考链接:https://www.cnblogs.com/karsa/p/13529769.html 这是CISCN2021 总决赛的题,解题思路是软链接zip 读取文件,然后伪造admin的 ...