Java 通过 SFTP 和 FTP 访问时相对路径引发的问题汇总

devtools/2024/9/23 20:15:36/

背景

常用的 Java SSH 操作工具包是 jsch ,FTP 工具包 commons-net ,本文将总结 Java 程序通过 SFTP 协议和 FTP 协议访问远程文件的过程中,需要注意的路径问题。

本文主要包括下面三个问题:

  1. FTPClient 获取当前用户根目录的方法
  2. ChannelSftp 获取当前用户根目录的方法。
  3. 目录切换时,如果是相对目录,可能存在的问题。
  4. 目标目录不存在时,逐级创建目录的方法。

相对路径操作注意事项

jsch 通过 ChannelSftp 访问 SFTP 时,获取当前登录用户根目录的方法有两种:

  1. pwd() 方法
  2. getHome() 方法

两者在刚刚登录成功后返回的值是一样的,都是当前登录用户的 home 目录。
FTPClient 获取当前登录用户的根目录方法为:printWorkingDirectory()

注意:如果操作的目标目录是相对路径,例如目标目录变量名称为 dir,判断绝对路径的方法为:

boolean isNotAbsolutePath = !dir.startWith(File.separator);
if (isNotAbsolutePath) {String homePath = ftpClient.printWorkingDirectory();或 String homePath = sftpChannel.pwd();// 最后拼接上根目录dir = homePath + File.separator + dir;
}

修正操作目标路径为绝对路径后,后面的操作就不会报文件不存在错误了。

listFiles 的内容

FTPClient 的 listFiles 文件列表中不包含 ... 这两个特殊的文件,但是使用 jsch 的 ChannelSftp 的 ls 直接列出的文件却包含 ...

所以在遍历文件过程中,对于 Sftp 访问操作,需要排除这两种特殊文件。

相对路径时 ls 或 cd 异常问题

如果使用相对路径,ChannelSftp 的 ls 操作会异常:

java">com.jcraft.jsch.SftpException: No such fileat com.jcraft.jsch.ChannelSftp.throwStatusError(ChannelSftp.java:2873) ~[jsch-0.1.54.jar:?]at com.jcraft.jsch.ChannelSftp._stat(ChannelSftp.java:2225) ~[jsch-0.1.54.jar:?]at com.jcraft.jsch.ChannelSftp._stat(ChannelSftp.java:2242) ~[jsch-0.1.54.jar:?]at com.jcraft.jsch.ChannelSftp.ls(ChannelSftp.java:1592) ~[jsch-0.1.54.jar:?]at com.jcraft.jsch.ChannelSftp.ls(ChannelSftp.java:1553) ~[jsch-0.1.54.jar:?]

解决办法,就是第二部分「相对路径操作注意事项」的内容,必须在真正切换目录之前拼接上当前登录用户的根目录。

逐级创建目录的方法

Java 的 FTP 和 SFTP 工具操作是的 mkdir 方法,当父级目录不存在时,直接用目录创建方法会失败,只能逐级创建每个目录。

确实有些场景下需要自动创建一个多层级的新目录,解决办法就是逐级拆解路径,判断当前路径是否存在,如果不存在,就创建,直到到达最后一级。

FTPClient 逐级创建一个目标目录 dirPath 的方法:

void createFilePath(FTPClient ftpClient, String filePath) {String[] dirPaths = filePath.split("/");for (int i = 0; i < dirPaths.length; i++) {if (StringUtils.isEmpty(dirPaths[i])) {continue;}StringBuffer buffer = new StringBuffer();for (int j = 0; j <= i; j++) {buffer.append(dirPaths[j]).append("/");}String tempDir = buffer.toString();try {// 切换目录,如果成功,则忽略boolean change= ftpClient.changeWorkingDirectory(filePath);if (change) {continue;}// 切换失败,则创建boolean result = ftpClient.makeDirectory(tempDir);} catch (Exception ex) {// ERROR}}
}

ChannelSftp 逐级创建一个目标目录 dirPath 的方法:

private boolean createFilePath(ChannelSftp channelSftp, String filePath) {String[] dirPaths = filePath.split("/");for (int i = 0; i < dirPaths.length; i++) {if (StringUtils.isEmpty(dirPaths[i])) {continue;}StringBuffer buffer = new StringBuffer();for (int j = 0; j <= i; j++) {buffer.append(dirPaths[j]).append("/");}String tempDir = buffer.toString();try {channelSftp.mkdir(tempDir);} catch (SftpException ex) {// ERROR}}
}

FTPClient 有 changeWorkingDirectory 方法返回一个 boolean 值,可以判断目录是否存在,不存在可以执行创建操作;ChannelSftp 的 cd 没有返回值,所以直接mkdir 逐级创建。


http://www.ppmy.cn/devtools/5181.html

相关文章

python中isinstance()作用

Python 中的 isinstance() 函数具有以下作用 &#xff08;类似Java 中的 instanceof &#xff09; 类型检查&#xff1a; isinstance() 用于判断一个对象是否属于特定的类型&#xff08;或类型集合&#xff09;。它接受两个参数&#xff1a;一个是待检测的对象&#xff0c;另一…

【Redis】string数据类型

文章目录 常用命令setsetnx & NXXXsetex & EXpsetex & PX msetget & mgetincr & decrincrby & decrbyincrbyfloatappendgetrangesetrangestrlen 内部编码 字符串类型是 Redis 最基础的数据类型。 在redis中所有的键都是 string 类型&#xff0c;其他的…

【C++】:函数重载,引用,内联函数,auto关键字,基于范围的for循环,nullptr关键字

目录 一&#xff0c;函数重载1.1 函数重载的定义1.1.1.形参的类型不同1.1.2参数的个数不同1.1.3.参数的顺序不同1.1.4.有一个是缺省参数构成重载。但是调用时存在歧义1.1.5.返回值不同&#xff0c;不构成重载。因为返回值可接收&#xff0c;可不接受&#xff0c;调用函数产生歧…

CentOS 7源码包与RPM包软件安装详解

CentOS 7源码包与RPM包软件安装详解 在CentOS 7中,软件包的安装主要有两种方式:源码包安装和RPM包安装。这两种方式各有优缺点,适用于不同的场景和需求。 一、源码包安装 源码包安装指的是从软件的源代码开始,通过编译过程将源代码转换为机器语言,进而实现软件的安装。…

Linux网络编程--网络传输

Linux网络编程--网络传输 Linux网络编程TCP/IP网络模型网络通信的过程局域网通信跨网络通信&#xff1a;问题总结&#xff1a; Linux网络编程 TCP/IP网络模型 发送方&#xff08;包装&#xff09;&#xff1a; 应用层&#xff1a;HTTP HTTPS SSH等 —> 包含数据&#xff0…

Java 集合(ArrayList、LinkedList、HashMap、HashSet、LinkedHashMap、LinkedHashSet)【补充复习】

Java 集合&#xff08;ArrayList、LinkedList、HashMap、HashSet、LinkedHashMap、LinkedHashSet&#xff09;【补充复习】 Java 集合概述Collection 接口继承树Map 接口继承树 Collection 接口方法使用 iterator 接口遍历集合元素使用 forearch 遍历集合元素 List 接口List 实…

小型燃气站3D可视化:打造安全高效的燃气新时代

随着科技的不断进步&#xff0c;越来越多的行业开始融入3D可视化技术&#xff0c;燃气行业也不例外。 小型燃气站作为城市燃气供应的重要节点&#xff0c;其安全性和运行效率至关重要。传统的燃气站管理方式往往依赖于人工巡检和纸质记录&#xff0c;这种方式不仅效率低下&…

C++11 列表初始化、initializer_list、auto、decltype、array

&#x1f436;博主主页&#xff1a;ᰔᩚ. 一怀明月ꦿ ❤️‍&#x1f525;专栏系列&#xff1a;线性代数&#xff0c;C初学者入门训练&#xff0c;题解C&#xff0c;C的使用文章&#xff0c;「初学」C &#x1f525;座右铭&#xff1a;“不要等到什么都没有了&#xff0c;才下…