最近在使用FTPClient连续读取ftp上的多个文件内容时,遇到了两个问题:

1. 在for循环中,FTPClient只能读取到第一个文件内容,读取第二个时遇到NPE问题。

2. 遇到程序锁死。

下面就遇到问题进行还原分析,并提供解决办法:

现在使用FTPClient读取ftp服务器/aaa/bbb/目录下a.txt,b.txt两个文件

 

 

 问题1:

 

 

 通过Debug发现在读取第一个文件前FTPClient的_replyCode=226(226 Transter complete.) 数据传输完的状态,当开始读第一个文件之后_replyCode=125(125 Data connection already open; Transter starting.)数据传输开始正在传输,表明第一文件还没有传输完,导致读第二个文件时inputStrean为null。

 

 

 

 

 

 

 解决办法是在读完第一文件之后,执行 ftpClient.completePendingCommand(); 告诉FTPClient数据已经传完。

问题2:

ftpClient.completePendingCommand();  在读取执行执行造成锁死。

解决办法 ftpClient.completePendingCommand(); 只能在数据传输之后执行。

最后贴上完整的代码:

 1 package com.shipc.demo.ftp;
 2 
 3 import org.apache.commons.net.ftp.FTPClient;
 4 import org.apache.commons.net.ftp.FTPFile;
 5 import org.apache.commons.net.ftp.FTPReply;
 6 
 7 import java.io.*;
 8 
 9 /**
10  * @author shipc 2019/10/24 10:00
11  * @version 1.0
12  */
13 public class TestFTP {
14 
15     private FTPClient connectServer() {
16         FTPClient ftpClient = new FTPClient();
17         ftpClient.setConnectTimeout(1000 * 60 * 30);
18         ftpClient.setControlEncoding("UTF-8");
19         ftpClient.enterLocalPassiveMode(); //设置被动模式,文件传输端口设置
20         try {
21             ftpClient.connect("172.17.0.16", 21);
22             ftpClient.login("ftp-user", "123456");
23             ftpClient.setFileType(FTPClient.BINARY_FILE_TYPE); // 设置文件传输模式为二进制
24             final int replyCode = ftpClient.getReplyCode();
25             if (!FTPReply.isPositiveCompletion(replyCode)) {
26                 ftpClient.disconnect();
27             }
28         } catch (Exception e) {
29             System.out.println(e.getMessage());
30         }
31         return ftpClient;
32     }
33     
34     private void readFiles(FTPClient ftpClient) throws IOException {
35         ftpClient.changeWorkingDirectory("/aaa/bbb");
36         final FTPFile[] ftpFiles = ftpClient.listFiles();
37         for (FTPFile ftpFile : ftpFiles) {
38             if (ftpFile.isFile()) {
39                 System.out.println("********文件" + ftpFile.getName() + "的内容如下**********");
40 //                ftpClient.completePendingCommand(); // 一定不能在读之前执行,否则会锁死
41                 
42                 InputStream inputStream = ftpClient.retrieveFileStream(ftpFile.getName());
43                 try (BufferedReader reader = new BufferedReader(new InputStreamReader(inputStream))) {
44                     while (true) {
45                         final String s = reader.readLine();
46                         if (s == null) {
47                             break;
48                         }
49                         System.out.println(s);
50                     }
51                 }
52                 inputStream.close();
53                 ftpClient.completePendingCommand(); // 每当读完一个文件时,要执行该语句
54             }
55         }
56     }
57 
58     public static void main(String[] args) {
59         final TestFTP testFTP = new TestFTP();
60         FTPClient ftpClient = testFTP.connectServer();
61         try {
62             testFTP.readFiles(ftpClient);
63         } catch (IOException e) {
64             e.printStackTrace();
65         }
66     }
67 }

 

版权声明:本文为shipc原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://www.cnblogs.com/shipc/p/shipc.html