JAVA FTPClient相关问题(速度慢,中文乱码,内容丢失)解决办法

FTPClient上传文件蜗牛速度的解决方法

Apache Commons的FTPClient局域网上传文件速度本应该很快的,但却在实际开发中发现上传一个文件蜗牛速度,都是因为调用了如下API:

ftpClient.storeFile(fileName, inputStream)

原因是因为默认缓冲区大小是1024,也就是1K,当然慢了,在调用上传API之前重新修改以下默认设置即可,如将缓冲区改为10M,API:

ftpClient.setBufferSize(1024 * 1024 * 10)

要在调用storeFile()前设置

通过FTPclient上传文件后,文件内容是乱码(文件被破坏问题)

用Apache的FTPClient上传文件时发现一个问题,就是上传txt文件没问题,但上传zip文件时文件会被破坏,查了一下原因,原来是这样:

因为RFC959中规定了缺省的传输模式应该是ASCII的,org.apache.commons.net.ftp.FTPClient实现也遵守此标准。所以org.apache.commons.net.ftp.FTPClient在缺省情况下是按ASCII形式进行传输的,如果你是传输的BINARY二进制文件(如zip),那么上传完后的文件就会被破坏,但是传输ASCII文件(如txt)是没有问题的。

所以如果你是传输的BINARY二进制文件的话,就需要在建立连接、登陆后,接下来设置文件类型,代码示例如下:

ftpclient.connect(host);
ftpclient.login(user, password);
ftpclient.setFileType(FTPClient.BINARY_FILE_TYPE);

注意顺序,设置类型要放在登录后,否则不生效

通过FTPclient上传文件后,中文文件名乱码

简单解决方法为:
将中文的目录或文件名转为iso-8859-1编码的字符。参考代码:

String name=“目录名或文件名”; name=new
String(name.getBytes(“GBK”),“iso-8859-1”);

很多人改为上述操作后,发现上传后中文不再乱码了,就以为解决了问题
还有人处理方法为:

ftpClient.setControlEncoding(“GBK”);
FTPClientConfig conf = new FTPClientConfig(FTPClientConfig.SYST_NT);
conf.setServerLanguageCode(“zh”);

该方法没有考虑ftp服务器的编码格式。本地搭建的Ftp服务器(windows2003 server)支持GBK编码方式,所以上述的解决方法可以,但是Ftp服务器(serv-u)是支持UTF-8格式的,所以此时在客户端的编码方式是GBK的,而搭设的ftp服务器中的设置就已经是utf-8的编码,所以肯定还是会出现乱码的问题。
可通过查看服务器是否支持UTF8选择使用什么编码

if (FTPReply.isPositiveCompletion(ftpClient.sendCommand( “OPTS UTF8”,“ON”))) {
// 开启服务器对UTF-8的支持,如果服务器支持就用UTF-8编码,否则就使用本地编码(GBK).
LOCAL_CHARSET = “UTF-8”; }

代码

import java.io.File;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.IOException;
import java.io.InputStream;
import java.io.UnsupportedEncodingException;
import org.apache.commons.net.ftp.FTPClient;
import org.apache.commons.net.ftp.FTPFile;
import org.apache.commons.net.ftp.FTPReply;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

/** 通过FTP上传文件 @Author lvhaibao @Date 2018/2/11 21:43 */
public class FTPTools {
    
    
  private static final Logger log = LoggerFactory.getLogger(FTPTools.class);

  /* 解决文件夹和文件名乱码*/
  /** 本地字符编码 */
  private static String LOCAL_CHARSET = "GBK";

  // FTP协议里面,规定文件名编码为iso-8859-1
  private static String SERVER_CHARSET = "ISO-8859-1";

  // 设置私有不能实例化
  private FTPTools() {
    
    }

  /**
   * 上传
   *
   * @param hostname
   * @param port
   * @param username
   * @param password
   * @param workingPath 服务器的工作目录
   * @param inputStream 文件的输入流
   * @param saveName 要保存的文件名
   * @return
   */
  public static boolean upload(
      String hostname,
      int port,
      String username,
      String password,
      String workingPath,
      InputStream inputStream,
      String saveName) {
    
    
    boolean flag = false;
    FTPClient ftpClient = new FTPClient();
    // 1 测试连接
    if (connect(ftpClient, hostname, port, username, password)) {
    
    
      try {
    
    
        // TODO 调试时查看该目录下文件
        FTPFile[] ftpFiles = ftpClient.listFiles(workingPath);
        for (FTPFile ftpFile : ftpFiles) {
    
    
          System.out.println(ftpFile.getName());
        }
        // 2 检查工作目录是否存在
        if (!ftpClient.changeWorkingDirectory(workingPath)) {
    
    
          // 如果目录不存在则创建目录
          // 因为ftp只支持一级级创建,固拆分后一级级创建
          String[] dirs = workingPath.split("/");
          String tempPath = "";
          for (String dir : dirs) {
    
    
            if (null == dir || "".equals(dir)) {
    
    
              continue;
            }
            tempPath += "/" + dir;
            if (!ftpClient.changeWorkingDirectory(tempPath)) {
    
    
              // ftp创建文件夹是否成功
              boolean b = ftpClient.makeDirectory(tempPath);
              if (!b) {
    
    
                return flag;
              } else {
    
    
                ftpClient.changeWorkingDirectory(tempPath);
              }
            }
          }
        }
        // 3 检查是否上传成功
        flag = storeFile(ftpClient, saveName, inputStream);
      } catch (IOException e) {
    
    
        log.error("工作目录不存在", e);
      } finally {
    
    
        disconnect(ftpClient);
      }
    }
    return flag;
  }

  /**
   * 断开连接
   *
   * @param ftpClient
   */
  public static void disconnect(FTPClient ftpClient) {
    
    
    if (ftpClient.isConnected()) {
    
    
      try {
    
    
        ftpClient.disconnect();
        log.info("已关闭连接");
      } catch (IOException e) {
    
    
        log.error("没有关闭连接", e);
      }
    }
  }

  /**
   * 测试是否能连接
   *
   * @param ftpClient
   * @param hostname ip或域名地址
   * @param port 端口
   * @param username 用户名
   * @param password 密码
   * @return 返回真则能连接
   */
  public static boolean connect(
      FTPClient ftpClient, String hostname, int port, String username, String password) {
    
    
    boolean flag = false;
    try {
    
    
      ftpClient.enterLocalPassiveMode();
      ftpClient.connect(hostname, port);
      // 开启服务器对UTF-8的支持,如果服务器支持就用UTF-8编码,否则就使用本地编码(GBK).
      if (FTPReply.isPositiveCompletion(ftpClient.sendCommand("OPTS UTF8", "ON"))) {
    
    
        LOCAL_CHARSET = "UTF-8";
      }
      ftpClient.setControlEncoding(LOCAL_CHARSET);
      if (ftpClient.login(username, password)) {
    
    
        // 设置文件类型必须放到登录后才生效,否则上传上去的文件内容依然乱码
        ftpClient.setFileType(FTPClient.BINARY_FILE_TYPE);
        log.info("连接ftp成功");
        flag = true;
      } else {
    
    
        disconnect(ftpClient);
        throw new RuntimeException("登录FTP失败,报告未生成!可能FTP用户名或密码错误!");
      }
    } catch (IOException e) {
    
    
      log.error("连接失败,可能ip或端口错误", e);
    }
    return flag;
  }

  /**
   * 上传文件
   *
   * @param ftpClient
   * @param saveName 全路径。如/home/public/a.txt
   * @param fileInputStream 输入的文件流
   * @return
   */
  public static boolean storeFile(
      FTPClient ftpClient, String saveName, InputStream fileInputStream) {
    
    
    boolean flag = false;
    try {
    
    
      ftpClient.setFileTransferMode(FTPClient.BINARY_FILE_TYPE);
      // 将缓存区变大,提升上传效果
      ftpClient.setBufferSize(1024 * 1024 * 10);
      flag = ftpClient.storeFile(saveName, fileInputStream);
      log.info("上传成功");
    } catch (IOException e) {
    
    
      log.error("上传失败", e);
    } finally {
    
    
      disconnect(ftpClient);
    }
    return flag;
  }

  public static void main(String[] args) throws FileNotFoundException {
    
    
    String hostname = "192.168.0.134";
    int port = 21;
    String username = "ceshi";
    String password = "test";
    String workingPath = "/test";
    String str = "C:\\Users\\T480S\\Desktop\\DF-5B-test-DF-5B-001-test.pdf";
    InputStream fileInputStream = new FileInputStream(new File(str));
        String saveName = "TS文档.docx";
    try {
    
    
      // 上传文件时,文件名称需要做编码转换
      saveName = new String(saveName.getBytes(LOCAL_CHARSET), SERVER_CHARSET);
    } catch (UnsupportedEncodingException e) {
    
    
      e.printStackTrace();
    }

    System.out.println(
        FTPTools.upload(
            hostname, port, username, password, workingPath, fileInputStream, saveName));
  }
}

猜你喜欢

转载自blog.csdn.net/qq_43961619/article/details/111317580