
本文介绍了如何使用 Java JSch 库在 SFTP 服务器上解压 ZIP 文件。通过创建独立的 SFTP 通道分别用于读取和写入操作,解决了 JSch 库在单个通道上同时处理多个文件流时可能出现的问题。同时,文章也提醒读者注意代码的实际行为,即先下载 ZIP 文件到本地解压,再将解压后的文件上传到 SFTP 服务器。
在使用 JSch 库处理 SFTP 服务器上的 ZIP 文件时,可能会遇到 java.io.IOException: error: 4: RequestQueue: unknown request id 错误。这通常是由于 JSch 库在单个 SFTP 通道上同时处理多个文件流时出现的问题。虽然理论上可以在一个 SFTP 通道上打开多个文件,但 JSch 库似乎对此支持不够完善。
解决方法:使用独立的 SFTP 通道
为了解决这个问题,可以为读取(get)操作和写入(put)操作分别创建独立的 SFTP 通道。 这样可以避免 JSch 库在单个通道上同时处理多个文件流时可能出现的冲突。
立即学习“Java免费学习笔记(深入)”;
以下是修改后的代码示例:
import com.jcraft.jsch.ChannelSftp;
import com.jcraft.jsch.JSch;
import com.jcraft.jsch.Session;
import com.jcraft.jsch.SftpException;
import com.jcraft.jsch.ChannelSftp.LsEntry;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.util.Vector;
import java.util.zip.ZipEntry;
import java.util.zip.ZipInputStream;
public class SftpUnzip {
public static void unzipSftpFile(String host, int port, String username, String password, String zipFilePath) throws Exception {
JSch jsch = new JSch();
Session session = jsch.getSession(username, host, port);
session.setPassword(password);
session.setConfig("StrictHostKeyChecking", "no"); // 忽略主机密钥检查,生产环境不建议这样做
session.connect();
ChannelSftp getChannel = (ChannelSftp) session.openChannel("sftp");
getChannel.connect();
ChannelSftp putChannel = (ChannelSftp) session.openChannel("sftp");
putChannel.connect();
try {
// 获取ZIP文件列表
Vector filelist = getChannel.ls(zipFilePath);
for (LsEntry entry : filelist) {
if (entry.getFilename().endsWith(".zip")) { // 确保只处理ZIP文件
try (
InputStream inputStream = getChannel.get(entry.getFilename());
ZipInputStream zipInputStream = new ZipInputStream(inputStream)
) {
ZipEntry zipEntry;
while ((zipEntry = zipInputStream.getNextEntry()) != null) {
if (!zipEntry.isDirectory()) { // 忽略目录
String entryName = zipEntry.getName();
try (OutputStream outputStream = putChannel.put(entryName)) {
byte[] buffer = new byte[2048];
int len;
while ((len = zipInputStream.read(buffer)) != -1) {
outputStream.write(buffer, 0, len);
}
} catch (SftpException e) {
System.err.println("Error writing file: " + entryName + " - " + e.getMessage());
}
}
zipInputStream.closeEntry();
}
} catch (IOException e) {
System.err.println("Error processing ZIP entry: " + e.getMessage());
}
}
}
} catch (SftpException e) {
System.err.println("Error listing or getting file: " + e.getMessage());
} finally {
if (getChannel != null && getChannel.isConnected()) {
getChannel.disconnect();
}
if (putChannel != null && putChannel.isConnected()) {
putChannel.disconnect();
}
if (session != null && session.isConnected()) {
session.disconnect();
}
}
}
public static void main(String[] args) {
String host = "your_sftp_host";
int port = 22; // 默认 SFTP 端口
String username = "your_username";
String password = "your_password";
String zipFilePath = "path/to/your/zip/files"; // ZIP 文件所在的路径
try {
unzipSftpFile(host, port, username, password, zipFilePath);
System.out.println("Unzip process completed.");
} catch (Exception e) {
System.err.println("An error occurred: " + e.getMessage());
e.printStackTrace();
}
}
} 代码解释:
良精商城网店购物系统是一套能够适合不同类型商品、超强灵活的多功能在线商店系统,三级分销 PC+移动端+微网站,为您提供了一个完整的在线开店解决方案。良精网店购物系统除了拥有一般网上商店系统所具有的所有功能,还拥有着其它网店系统没有的许多超强功能。多种独创的技术使得系统能满足各行业广大用户的各种各样的需求,是一个经过完善设计并适用于各种服务器环境的高效、全新、快速和优秀的网上购物软件解决方案。
- 创建独立的 SFTP 通道: getChannel 用于从 SFTP 服务器读取 ZIP 文件,putChannel 用于将解压后的文件上传到 SFTP 服务器。
- 使用 try-with-resources 语句: 确保在操作完成后关闭 InputStream 和 OutputStream,避免资源泄漏。
- 异常处理: 添加了更完善的异常处理,以便在出现错误时能够更好地诊断问题。
- 检查文件类型: 确保只处理 ZIP 文件。
- 忽略目录: 忽略 ZIP 文件中的目录条目。
注意事项:
- 主机密钥验证: 示例代码中禁用了主机密钥验证 (StrictHostKeyChecking=no)。在生产环境中,强烈建议配置正确的主机密钥验证,以提高安全性。
- 目录结构: 上述代码会将 ZIP 文件中的所有文件解压到 SFTP 服务器的当前目录下。如果需要保留 ZIP 文件中的目录结构,需要修改代码来创建相应的目录。
- 文件覆盖: 如果解压后的文件名与 SFTP 服务器上已存在的文件名冲突,上述代码会覆盖已存在的文件。需要根据实际需求进行修改。
- 安全性: 请务必保护好 SFTP 服务器的用户名和密码,避免泄露。
总结:
虽然上述代码可以解决 JSch 库在单个通道上同时处理多个文件流时可能出现的问题,但需要注意的是,它实际上是将 ZIP 文件下载到本地解压,然后再将解压后的文件上传到 SFTP 服务器。如果 ZIP 文件非常大,这可能会消耗大量的本地资源和网络带宽。
如果需要直接在 SFTP 服务器上解压 ZIP 文件,可能需要考虑使用其他工具或方法,例如:
- 在 SFTP 服务器上安装 unzip 工具: 然后使用 JSch 库执行 shell 命令来解压 ZIP 文件。
- 使用其他支持直接在远程服务器上解压 ZIP 文件的库: 例如,某些商业 SFTP 客户端库可能提供此功能。
选择哪种方法取决于具体的应用场景和需求。









