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));
}
}