Java通過FTPClient實現登陸、文件上傳、文件下載以及文件夾的遍歷

最近工作中遇到需要定期從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);

 

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章