2019独角兽企业重金招聘Python工程师标准>>>
这几天在做一个FTP上传文件的功能,在本地运行完全没有问题,上传文件成功,图片成功显示。在公网(centos系统)上进行测试的时候就出了问题了,上传一直卡到storeFile上,一般是会卡住30-60秒,然后返回“false”,不报异常。所以我又开始怀疑代码的问题,在本地建了个CentOS的虚拟机,安装vsftp,结果是成功的。使用FTP上传工具进行处理公网上传下载一点问题都没有,不会出现图片不显示以及文件0字节的问题,至此,我就十分无耐了,同样的代码,在不同环境上支行的效果居然不一样。
具体代码如下:
=====================上传文件的调用===============================
public String upload(MultipartFile file, String path) {
String fileName = file.getOriginalFilename();
//扩展名
//abc.jpg
String fileExtensionName = fileName.substring(fileName.lastIndexOf(".") + 1);
String uploadFileName = UUID.randomUUID().toString() + "." + fileExtensionName;
log.info("开始上传文件,上传文件的文件名:{},上传的路径:{},新文件名:{}", fileName, path, uploadFileName);
File fileDir = new File(path);
if (!fileDir.exists()) {
fileDir.setWritable(true);
fileDir.mkdirs();
}
File targetFile = new File(path, uploadFileName);
try {
file.transferTo(targetFile);
//文件已经上传成功了
log.info("文件已经上传成功了,上传文件的文件名:{},上传的路径:{},新文件名:{}", fileName, path, uploadFileName);
FTPUtil.uploadFile(Lists.newArrayList(targetFile));
//已经上传到ftp服务器上
log.info("已经上传到ftp服务器上,上传文件的文件名:{},上传的路径:{},新文件名:{}", fileName, path, uploadFileName);
targetFile.delete();
}catch (IOException e) {
log.error("上传文件异常", e);
return null;
}
return targetFile.getName();
}
==========================封装成工具使用===========================
public class FTPUtil {
private static String ftpIp = PropertiesUtil.getProperty("ftp.server.ip");
private static String ftpUser = PropertiesUtil.getProperty("ftp.user");
private static String ftpPass = PropertiesUtil.getProperty("ftp.pass");
public FTPUtil(String ip,int port,String user,String pwd){
this.ip = ip;
this.port = port;
this.user = user;
this.pwd = pwd;
}
public static boolean uploadFile(ListfileList) throws IOException {
FTPUtil ftpUtil = new FTPUtil(ftpIp,21,ftpUser,ftpPass);
log.info("开始连接ftp服务器");
boolean result = ftpUtil.uploadFile("img",fileList);
log.info("开始连接ftp服务器,结束上传,上传结果:{}, fileList:{}", result,fileList.get(0).getName());
return result;
}
private boolean uploadFile(String remotePath,ListfileList) throws IOException {
boolean uploaded = true;
FileInputStream fis = null;
//连接FTP服务器
if(connectServer(this.ip,this.port,this.user,this.pwd)){
try {
ftpClient.changeWorkingDirectory(remotePath);
ftpClient.setBufferSize(1024);
ftpClient.setControlEncoding("UTF-8");
ftpClient.setFileType(FTPClient.BINARY_FILE_TYPE);
// 开启被动模式,不建议开启。本地测试可以不开启,服务端需要开启
ftpClient.enterLocalPassiveMode();
for(File fileItem : fileList){
fis = new FileInputStream(fileItem);
ftpClient.storeFile(fileItem.getName(),fis);
log.info("============文件名,{}========", fileItem.getName());
}} catch (IOException e) {
log.error("上传文件异常",e);
uploaded = false;
e.printStackTrace();
} finally {
fis.close();
ftpClient.disconnect();
}
}
log.info("=====uploaded:{}", uploaded);
return uploaded;
}
private boolean connectServer(String ip,int port,String user,String pwd){
boolean isSuccess = false;
ftpClient = new FTPClient();
try {
ftpClient.connect(ip);
isSuccess = ftpClient.login(user,pwd);
} catch (IOException e) {
log.error("连接FTP服务器异常",e);
}
return isSuccess;
}
private String ip;
private int port;
private String user;
private String pwd;
private FTPClient ftpClient;
...省略get()...set()...
}
原因:
是客户端上传和服务端上传的模式不一样导致的。客户端上传可以使用主动模式上传,服务端需要使用被动模式。
解决方案:
ftp开放被动模式,防火墙开放被动模式的端口号(比如30000-30999),代码上增加一行
// 开启被动模式,不建议开启。本地测试可以不开启,服务端需要开启
ftpClient.enterLocalPassiveMode();