共计 2250 个字符,预计需要花费 6 分钟才能阅读完成。
本文 FTP 连接的相关操作均在 被动模式 下进行。
FTP 的端口分两种:控制端口 和 数据端口。
连接 FTP 服务器时的 port 的默认 21 为控制端口。
FTP 的数据端口顾名思义就是用来进行数据操作的端口。其又分为两种模式:主动模式 与 被动模式。
主动模式端口默认为 20,被动模式端口为范围 3000/4000。
当服务器上存在防火墙时,需注意开放相关端口的问题。
完整示例代码请见最后相关下载部分。
连接服务器并登录
public FTPClient connect(String hostname, Integer port, String username, String password) throws IOException { | |
// 创建连接 | |
FTPClient ftpClient = new FTPClient(); | |
ftpClient.connect(hostname, port); | |
// 设置用于 FTP 控制连接的编码:UTF- 8 等 | |
ftpClient.setControlEncoding(properties.getEncoding()); | |
log.error("windcoder.com:connect=>\t"+ftpClient.getReplyString()); | |
// 登录服务器 | |
ftpClient.login(username, password); | |
log.error("login=>\t"+ftpClient.getReplyString()); | |
return ftpClient; | |
} |
退出并断开连接
public void disconnect(FTPClient ftpClient) throws IOException {ftpClient.logout(); | |
log.error("disLogin=>\t"+ftpClient.getReplyString()); | |
ftpClient.disconnect(); | |
log.error("disConnect=>\t"+ftpClient.getReplyString()); | |
} |
获取并下载文件关键代码
// 进入文件所在远程目录 | |
ftpClient.changeWorkingDirectory(remoteFilePath); | |
// 开启被动模式 | |
ftpClient.enterLocalPassiveMode(); | |
// 设置以二进制方式传输 | |
ftpClient.setFileType(FTPClient.BINARY_FILE_TYPE); | |
// 获取文件,listFiles 参数为空时获取当前目录下所有文件,包含文件名时,可通过 files.length == 1 判断是否存在该文件 | |
FTPFile[] files = ftpClient.listFiles(tmpFileName); | |
// 获取文件大小 | |
long remoteSize = files[0].getSize(); | |
// 返回一个用于读取服务器上名为 tmpFileName 的文件的 InputStream,InputStream in = ftpClient.retrieveFileStream(tmpFileName); | |
// 创建一个用于写入 localFile 文件的输出流,若需追加写,则添加并将第二个参数设为 true。FileOutputStream out = new FileOutputStream(localFile, true); | |
// 将 in 复制到 out,具体方式可自由实现,这里展示 IOUtils 方式 | |
if (remoteSize - localSize >= 2 * FileUtils.ONE_GB) { | |
// 文件大小超过 2G 时 | |
IOUtils.copyLarge(in, out); | |
}else {IOUtils.copy(in, out); | |
} | |
// 刷新此输出流,并强制写出所有缓冲的输出字节。flush 的一般约定是,调用它表明,如果先前写入的任何字节已由输出流的实现缓冲,则应立即将这些字节写入其预期的目标。out.flush(); | |
// 关闭相关流,这里展示 IOUtils 方式 | |
IOUtils.closeQuietly(in); | |
IOUtils.closeQuietly(out); | |
// 接收来自服务器的完成答复并验证整个事务是否成功 | |
/** | |
* 有些 FTPClient 方法不能完成整个 FTP 命令序列来完成事务。这些命令要求程序员在收到肯定的中间命令后采取一些措施。* 程序员的代码完成其操作后,必须调用此方法以接收来自服务器的完成答复并验证整个事务是否成功。* */ | |
ftpClient.completePendingCommand(); |
获取并上传文件关键代码
// 创建输入流 | |
FileInputStream fis = new FileInputStream(filePath); | |
// 进入文件所在 FTP 远程目录 | |
ftpClient.changeWorkingDirectory(uploadPath); | |
// 开启被动模式 | |
ftpClient.enterLocalPassiveMode(); | |
// 调整 ftp 传输模式为二进制方式 | |
ftpClient.setFileType(FTPClient.BINARY_FILE_TYPE); | |
// 调用 ftp 的方法上载 | |
// 从给定 InputStream 中获取输入并以给定文件名 fileName 将文件保存在 FTP 服务器上。boolean ret = ftpClient.storeFile(fileName, fis); | |
// 关闭文件流 | |
fis.close(); |
参考资料
FtpConnection.java
本文首发 Windcoder。
正文完
发表至: java
2019-10-22