问题描述
- 笔者最近遇到一个需求
- 就是把后端接口返回的zip压缩包中的文件提取出来
- 并且解析拿到压缩包中的文件具体内容
- (比如,如果是txt文件,要读取其中的txt里面有哪些文字)
- 下载其中部分文件使用file-saver的saveAs方法
Nodejs的express提供一个接口返回一个zip压缩包
为了方便更好阅读,笔者用express提供了一个接口
// 返回一个zip压缩包
route.get('/getZip', (req, res) => {
// 提前存储一份zip文件的路径
let zipUrl = './public/zip.zip'
// 此接口允许跨域
res.header('Access-Control-Allow-Origin', '*');
// 设置请求头(指定类型为zip)
res.setHeader('Content-Type', 'application/zip');
res.setHeader('Content-Disposition', 'attachment; filename=zip.zip');
//创建可读流 引入文件模块 const fs = require("fs")
let readStream = fs.createReadStream(zipUrl)
// 将读取的结果以管道pipe流的方式返回给前端
readStream.pipe(res);
})
- 这样的话,我们就可以给这个接口发送请求,获取对应的压缩包文件了
- 接口地址:http://ashuai.work/api/getZip
- 当然,大家也可以复制粘贴到浏览器地址栏中回车直接下载
- zip压缩包内容如下:
下载jszip和file-saver
npm i jszip file-saver --save
"jszip": "^3.10.1",
"file-saver": "^2.0.5",
请求压缩包数据、并解压、再读取文件
比如,我是点击按钮请求数据
<template>
<div>
<button @click="downloadZipAndExtract">下载并提取压缩文件中的内容文件</button>
</div>
</template>
<script setup>
import { saveAs } from 'file-saver';
import JSZip from 'jszip';
import axios from 'axios';
const URL = 'http://ashuai.work/api/getZip'
const downloadZipAndExtract = () => {
// 发请求,获取zip压缩包
axios.get(URL, { responseType: 'blob' })
.then(async response => {
const blob = response.data;
console.log('blob', blob)
})
.catch(error => {
console.error('下载失败', error);
});
};
</script>
上述打印结果如下图:
size是文件大小,289759字节大小;类型为zip
把接口返回的blob,交给zip去异步加载
const downloadZipAndExtract = () => {
// 发请求,获取zip压缩包
axios.get(URL, { responseType: 'blob' })
.then(async response => {
const blob = response.data;
// 实例化zip压缩包
const zip = new JSZip();
// 加载zip压缩包的内容
await zip.loadAsync(blob);
console.log('zip', zip)
})
.catch(error => {
console.error('下载失败', error);
});
};
上述打印结果如下:我们已经可以看到zip压缩包中的文件有哪些了
遍历zip中的files,既可拿到对应文件,再使用FileReader读取内容
// 接口返回的blob数据
const blob = response.data;
// 实例化zip压缩包
const zip = new JSZip();
// 加载zip压缩包的内容
await zip.loadAsync(blob);
// 遍历zip压缩包中的内容
zip.forEach((relativePath, file) => {
// 不读文件夹,读取文件
if (!file.dir) {
// 读取文件
file.async('blob').then(content => {
// 使用FileReader 读取文件内容
const reader = new FileReader();
reader.onload = (e) => {
console.log('文件内容------>', e.target.result);
}
reader.readAsText(content);
// 保存文件
saveAs(content, file.name);
})
}
});
上述打印文件内容如下:
完整代码
复制粘贴即用
<template>
<div>
<button @click="downloadZipAndExtract">下载并提取压缩文件中的内容文件</button>
</div>
</template>
<script setup>
import { saveAs } from 'file-saver';
import JSZip from 'jszip';
import axios from 'axios';
const URL = 'http://ashuai.work/api/getZip'
const downloadZipAndExtract = () => {
// 发请求,获取zip压缩包
axios.get(URL, { responseType: 'blob' })
.then(async response => {
const blob = response.data;
// 实例化zip压缩包
const zip = new JSZip();
// 加载zip压缩包的内容
await zip.loadAsync(blob);
// 遍历zip压缩包中的内容
zip.forEach((relativePath, file) => {
// 不读文件夹,读取文件
if (!file.dir) {
// 读取文件
file.async('blob').then(content => {
// 使用FileReader 读取文件内容
const reader = new FileReader();
reader.onload = (e) => {
console.log('文件内容------>', e.target.result);
}
reader.readAsText(content);
// 保存文件
saveAs(content, file.name);
})
}
});
})
.catch(error => {
console.error('下载失败', error);
});
};
</script>
地址:
jszip在线压缩
- jszip除了能够解压zip压缩包中文件之外
- 还能够把文件(文件夹)压缩成zip包
- 如下的html文件
- 复制粘贴即用
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>zip压缩文件和文件夹</title>
<script src="https://cdnjs.cloudflare.com/ajax/libs/jszip/3.10.1/jszip.min.js"></script>
</head>
<body>
<h1>文件和文件夹压缩</h1>
<button id="compressFiles">压缩文件</button>
<button id="compressFolder">压缩文件夹</button>
<input type="file" id="fileInput" multiple style="display: none;">
<input type="file" id="folderInput" webkitdirectory directory style="display: none;">
<script>
// 压缩文件按钮
const compressFilesButton = document.getElementById('compressFiles');
// 压缩文件夹按钮
const compressFolderButton = document.getElementById('compressFolder');
// 文件框子
const fileInput = document.getElementById('fileInput');
// 文件夹框子
const folderInput = document.getElementById('folderInput');
// 派发文件框子点击
compressFilesButton.addEventListener('click', () => {
fileInput.click();
});
// 派发文件夹框子点击
compressFolderButton.addEventListener('click', () => {
folderInput.click();
});
// 上传文件,就去压缩
fileInput.addEventListener('change', async () => {
const files = fileInput.files;
if (files.length > 0) {
await compressAndDownload(files, 'compressed_files.zip');
}
});
// 上传文件夹,就去压缩
folderInput.addEventListener('change', async () => {
const files = folderInput.files;
if (files.length > 0) {
await compressAndDownload(files, 'compressed_folder.zip');
}
});
// 统一压缩并下载逻辑
async function compressAndDownload(files, zipFileName) {
// 实例化
const zip = new JSZip();
for (let i = 0; i < files.length; i++) {
const file = files[i];
const relativePath = file.webkitRelativePath || file.name;
// zip装填文件
zip.file(relativePath, file);
}
// 把装填好的文件,压缩
const content = await zip.generateAsync({ type: 'blob' });
// 下载逻辑
const link = document.createElement('a');
link.href = URL.createObjectURL(content);
link.download = zipFileName;
link.click();
URL.revokeObjectURL(link.href);
}
</script>
</body>
</html>
**粗体** _斜体_ [链接](http://example.com) `代码` - 列表 > 引用
。你还可以使用@
来通知其他用户。