最近工作中遇到需要定期從FTP上下載文件的需求,故整理了一個工具類以供後續使用。
工具類中主要使用了org.apache.commons.net.ftp中的相關類,因此加入如下的Maven依賴:
<!-- https://mvnrepository.com/artifact/commons-net/commons-net -->
<dependency>
<groupId>commons-net</groupId>
<artifactId>commons-net</artifactId>
<version>3.6</version>
</dependency>
創建FTP工具類:FTPUtil.java。具體使用方法詳見註釋:
import org.apache.commons.net.ftp.FTPClient;
import org.apache.commons.net.ftp.FTPFile;
import org.apache.commons.net.ftp.FTPReply;
import org.apache.log4j.Logger;
import java.io.*;
public class FTPUtil {
private static Logger logger = Logger.getLogger(FTPUtil.class);
/**
* 登陸FTP並獲取FTPClient對象
*
* @param host FTP主機地址
* @param port FTP端口
* @param userName 登錄用戶名
* @param password 登錄密碼
* @return
*/
public static FTPClient loginFTP(String host, int port, String userName, String password) {
FTPClient ftpClient = null;
try {
ftpClient = new FTPClient();
// 連接FTP服務器
ftpClient.connect(host, port);
// 登陸FTP服務器
ftpClient.login(userName, password);
// 中文支持
ftpClient.setControlEncoding("UTF-8");
// 設置文件類型爲二進制(如果從FTP下載或上傳的文件是壓縮文件的時候,不進行該設置可能會導致獲取的壓縮文件解壓失敗)
ftpClient.setFileType(FTPClient.BINARY_FILE_TYPE);
ftpClient.enterLocalPassiveMode();
if (!FTPReply.isPositiveCompletion(ftpClient.getReplyCode())) {
logger.error("連接FTP失敗,用戶名或密碼錯誤。");
ftpClient.disconnect();
} else {
logger.info("FTP連接成功!");
}
} catch (Exception e) {
logger.error("登陸FTP失敗,請檢查FTP相關配置信息是否正確!", e);
}
return ftpClient;
}
/**
* 從FTP下載文件到本地
* @param ftpClient 已經登陸成功的FTPClient
* @param ftpFilePath FTP上的目標文件路徑
* @param localFilePath 下載到本地的文件路徑
*/
public static void downloadFileFromFTP(FTPClient ftpClient, String ftpFilePath, String localFilePath) {
InputStream is = null;
FileOutputStream fos = null;
try {
// 獲取ftp上的文件
is = ftpClient.retrieveFileStream(ftpFilePath);
fos = new FileOutputStream(new File(localFilePath));
// 文件讀取方式一
int i;
byte[] bytes = new byte[1024];
while ((i = is.read(bytes)) != -1) {
fos.write(bytes, 0, i);
}
// 文件讀取方式二
//ftpClient.retrieveFile(ftpFilePath, new FileOutputStream(new File(localFilePath)));
ftpClient.completePendingCommand();
logger.info("FTP文件下載成功!");
} catch (Exception e) {
logger.error("FTP文件下載失敗!", e);
} finally {
try {
if (fos != null) {
fos.close();
}
if (is != null) {
is.close();
}
} catch (IOException e) {
e.printStackTrace();
}
}
}
/**
* 上傳本地文件到FTP
* @param ftpClient 已經登陸成功的FTPClient
* @param localFilePath 要上傳的本地文件路徑
* @param ftpFilePath 要保存的FTP文件路徑
*/
public static void uploadFileToFTP(FTPClient ftpClient, String localFilePath, String ftpFilePath) {
OutputStream os = null;
FileInputStream fis = null;
try {
// 獲取ftp上的文件
os = ftpClient.storeFileStream(ftpFilePath);
fis = new FileInputStream(new File(localFilePath));
// 文件保存方式一
int length;
byte[] bytes = new byte[1024];
while ((length = fis.read(bytes)) != -1) {
os.write(bytes, 0, length);
}
// 文件保存方式二
//ftpClient.storeFile(ftpFilePath, new FileInputStream(new File(localFilePath)));
ftpClient.completePendingCommand();
logger.info("FTP文件上傳成功!");
} catch (Exception e) {
logger.error("FTP文件上傳失敗!", e);
} finally {
try {
if (fis != null) {
fis.close();
}
if (os != null) {
os.close();
}
} catch (IOException e) {
e.printStackTrace();
}
}
}
/**
* 獲取FTP某一特定目錄下的所有文件名稱
* @param ftpClient 已經登陸成功的FTPClient
* @param ftpDirPath FTP上的目標文件路徑
*/
public void getFileNameListFromFTP(FTPClient ftpClient, String ftpDirPath) {
try {
if (ftpDirPath.startsWith("/") && ftpDirPath.endsWith("/")) {
// 通過提供的文件路徑獲取FTPFile對象列表
FTPFile[] files = ftpClient.listFiles(ftpDirPath);
// 遍歷文件列表,打印出文件名稱
for (int i = 0; i < files.length; i++) {
FTPFile ftpFile = files[i];
// 此處只打印文件,未遍歷子目錄(如果需要遍歷,加上遞歸邏輯即可)
if (ftpFile.isFile()) {
logger.info(ftpDirPath + ftpFile.getName());
}
}
} else {
logger.error("當前FTP路徑不可用");
}
} catch (IOException e) {
e.printStackTrace();
}
}
}
最後創建測試類FTPUtilTest.java:
import org.apache.commons.net.ftp.FTPClient;
import java.io.IOException;
public class FTPUtilTest {
public static void main(String[] args) throws IOException {
FTPClient ftpClient = FTPUtil.loginFTP("192.168.8.88", 21, "userName", "******");
FTPUtil.downloadFileFromFTP(ftpClient, "/usr/local/test.txt", "/Users/123/Downloads/test.txt");
FTPUtil.uploadFileToFTP(ftpClient, "/Users/123/Downloads/test2.txt", "/usr/local/test2.txt");
ftpClient.disconnect();
}
}
注意事項:
一、很多人在循環調用上傳或下載方法的時候,有的會遇到第一次調用沒問題,但是從第二次開始ftpClient.retrieveFileStream和ftpClient.storeFileStream返回null的問題,因此此問題的原因在於漏掉了下面這句代碼
ftpClient.completePendingCommand();
切記在成功讀取或者上傳一個文件之後調用該方法。
二、如果遇到上傳或者下載後的Zip文件無法打開或者使用密碼無法解壓的情況,則是遺漏了下面這句代碼:
ftpClient.setFileType(FTPClient.BINARY_FILE_TYPE);