基于FTP4J组件的FTP操作客户端,支持上传、下载,两台FTP服务器文件流同步。
package com.matol.utils
import it.sauronsoftware.ftp4j.FTPClient;
import it.sauronsoftware.ftp4j.FTPDataTransferListener;
import it.sauronsoftware.ftp4j.FTPFile;
import java.io.File;
import java.io.FileInputStream;
import java.io.InputStream;
import java.util.ArrayList;
import java.util.List;
/**
* 基于FTP4J组件的FTP操作客户端
* ww
*/
public final class FTPToolkit {
//单例模型
private FTPToolkit() { }
/**
* 创建FTP连接
* @param host 主机名或IP
* @param port ftp端口
* @param username ftp用户名
* @param password ftp密码
* @return 一个单例客户端
*/
public static FTPClient ftpConn(String host, int port, String username, String password) {
FTPClient client = new FTPClient();
try {
client.connect(host, port);
client.login(username, password);
client.setCharset("GBK");
client.setType(FTPClient.TYPE_BINARY);
} catch (Exception e) {
System.out.println("###[Error] FTPToolkit.ftpConn() "+e.getMessage());
}
return client;
}
/**
* 取得目录下所有文件列表
* @param path
* @return
*/
public static List<FTPFile> getlistFiles(FTPClient client,String path) {
List<FTPFile> filesList = new ArrayList<FTPFile>();
try {
client.changeDirectory(path);
FTPFile[] fileNames = client.list(); //.listNames();
if (null != fileNames) {
for (FTPFile file : fileNames) {
if(file.getType() == FTPFile.TYPE_FILE){
filesList.add(file);
//System.out.println(file.getName()+" | "+file.getSize()+" | "+DateUtil.converDateToString(file.getModifiedDate(),DateUtil.defaultDatePattern));
}
}
}
} catch (Exception e) {
System.out.println("###[Error] FTPToolkit.getlistFiles()"+ e.getMessage());
}
return filesList;
}
/**
* 取得目录下所有文件列表
* @param path
* @return
*/
public static List<String> getlistFileNames(FTPClient client,String path) {
List<String> filesList = new ArrayList<String>();
try {
client.changeDirectory(path);
String[] fileNames = client.listNames();
if (null != fileNames) {
for (String file : fileNames) {
filesList.add(path+"/"+file);
//System.out.println(path+"/"+file);
}
}
} catch (Exception e) {
System.out.println("###[Error] FTPToolkit.getlistFileNames()"+ e.getMessage());
}
return filesList;
}
/**
* 根据文件路径获取文件流
* @param path
* @return
*/
public static InputStream fetchInputStream(FTPClient client,String ftpFilePath,String ftpPath) {
InputStream is = null;
try {
String localFilePath = getFileName("ftptmp/"+ftpFilePath) + "_ftp.tmp";
File tempLocalFile = new File(localFilePath);
if (!tempLocalFile.exists()) {
tempLocalFile.createNewFile();
}
client.download(ftpPath, tempLocalFile);
is = new FileInputStream(tempLocalFile);
//删除临时文件(由于不能关闭文件流,删除临时文件无效,所以最好使用deleteOnExit,外层程序使用文件流结束后,关闭流,自动删除)
//tempLocalFile.deleteOnExit();
} catch (Exception e) {
System.out.println("###[Error] FTPToolkit.fetchInputStream()"+e.getMessage());
}
return is;
}
/**
* 上传文件到FTP
* @param ftpFilePath 要上传到FTP的路径
* @param source 源数据
*/
public static void uploadInputStream(FTPClient client,String ftpFilePath, InputStream ftpFile) {
try {
mkDirs(client,ftpFilePath.substring(0, ftpFilePath.lastIndexOf("/")));
client.upload(ftpFilePath, ftpFile, 0l, 0l, null);
} catch (Exception e) {
System.out.println("###[Error] FTPToolkit.uploadInputStream()"+e.getMessage());
}finally{
try {
//关闭文件流
ftpFile.close();
} catch (Exception e) {}
}
}
/**
* FTP下载文件到本地一个文件夹,如果本地文件夹不存在,则创建必要的目录结构
* @param client FTP客户端
* @param remoteFileName FTP文件
* @param localFolderPath 存的本地目录
*/
public static void download(FTPClient client, String remoteFileName, String localFolderPath) {
try {
int x = isExist(client, remoteFileName);
MyFtpListener listener = MyFtpListener.instance("download");
File localFolder = new File(localFolderPath);
if (!localFolder.exists()) {
localFolder.mkdirs();
}
if (x == FTPFile.TYPE_FILE) {
String localfilepath = formatPath4File(localFolderPath+File.separator+new File(remoteFileName).getName());
if (listener != null) {
client.download(remoteFileName, new File(localfilepath), listener);
}else{
client.download(remoteFileName, new File(localfilepath));
}
}
} catch (Exception e) {
System.out.println("###[Error] FTPToolkit.download()"+e.getMessage());
}
}
//递归创建层级目录(坑爹的问题,原本想着递归处理,后来才发现如此的简单)
private static void mkDirs(FTPClient client,String p) throws Exception {
if (null == p) { return;}
if(p != null && !"".equals(p) && !"/".equals(p)){
String ps = "";
for(int i=0;i<p.split("/").length;i++){
ps += p.split("/")[i]+"/";
if (!isDirExist(client,ps)) {
client.createDirectory(ps);// 创建目录
client.changeDirectory(ps);// 进入创建的目录
System.out.println(">>>>> create directory:["+i+"]["+ps+"]");
}else{
//System.out.println("select directory:["+i+"]["+ps+"]");
}
}
}
}
//检查目录是否存在
private static boolean isDirExist(FTPClient client,String dir) {
try {
client.changeDirectory(dir);
} catch (Exception e) {
return false;
}
return true;
}
/**
* FTP上传本地文件到FTP的一个目录下
* @param client FTP客户端
* @param localfilepath 本地文件路径
* @param remoteFolderPath FTP上传目录
*/
public static void upload(FTPClient client, String localfilepath, String remoteFolderPath) {
try {
mkDirs(client,remoteFolderPath);
File localfile = new File(localfilepath);
upload(client, localfile, remoteFolderPath);
} catch (Exception e) {
System.out.println("###[Error] FTPToolkit.upload()"+e.getMessage());
}
}
/**
* FTP上传本地文件到FTP的一个目录下
* @param client FTP客户端
* @param localfile 本地文件
* @param remoteFolderPath FTP上传目录
*/
public static void upload(FTPClient client, File localfile, String remoteFolderPath) {
remoteFolderPath = formatPath4FTP(remoteFolderPath);
MyFtpListener listener = MyFtpListener.instance("upload");
try {
client.changeDirectory(remoteFolderPath);
if (listener != null) {
client.upload(localfile, listener);
}else{
client.upload(localfile);
}
client.changeDirectory("/");
} catch (Exception e) {
System.out.println("###[Error] FTPToolkit.upload()"+e.getMessage());
}
}
/**
* 批量上传本地文件到FTP指定目录上
* @param client FTP客户端
* @param localFilePaths 本地文件路径列表
* @param remoteFolderPath FTP上传目录
*/
public static void uploadListPath(FTPClient client, List<String> localFilePaths, String remoteFolderPath) {
try {
remoteFolderPath = formatPath4FTP(remoteFolderPath);
client.changeDirectory(remoteFolderPath);
MyFtpListener listener = MyFtpListener.instance("uploadListPath");
for (String path : localFilePaths) {
File file = new File(path);
if (listener != null) {
client.upload(file, listener);
}else {
client.upload(file);
}
}
client.changeDirectory("/");
} catch (Exception e) {
System.out.println("###[Error] FTPToolkit.uploadListPath()"+e.getMessage());
}
}
/**
* 批量上传本地文件到FTP指定目录上
* @param client FTP客户端
* @param localFiles 本地文件列表
* @param remoteFolderPath FTP上传目录
*/
public static void uploadListFile(FTPClient client, List<File> localFiles, String remoteFolderPath) {
try {
client.changeDirectory(remoteFolderPath);
remoteFolderPath = formatPath4FTP(remoteFolderPath);
MyFtpListener listener = MyFtpListener.instance("uploadListFile");
for (File file : localFiles) {
if (listener != null){
client.upload(file, listener);
}else{
client.upload(file);
}
}
client.changeDirectory("/");
} catch (Exception e) {
System.out.println("###[Error] FTPToolkit.uploadListFile() "+e.getMessage());
}
}
/**
* 判断一个FTP路径是否存在,如果存在返回类型(FTPFile.TYPE_DIRECTORY=1、FTPFile.TYPE_FILE=0、FTPFile.TYPE_LINK=2)
* @param client FTP客户端
* @param remotePath FTP文件或文件夹路径
* @return 存在时候返回类型值(文件0,文件夹1,连接2),不存在则返回-1
*/
public static int isExist(FTPClient client, String remotePath) {
int x = -1;
try {
remotePath = formatPath4FTP(remotePath);
FTPFile[] list = client.list(remotePath);
if (list.length > 1){
x = 1;
}else if (list.length == 1) {
FTPFile f = list[0];
if (f.getType() == FTPFile.TYPE_DIRECTORY) {
x = 1;
}
//假设推理判断
String _path = remotePath + "/" + f.getName();
if(client.list(_path).length == 1){
x = 1;
}else{
x = 0;
}
} else {
client.changeDirectory(remotePath);
x = 1;
}
} catch (Exception e) {
x = -1;
System.out.println("###[Error] FTPToolkit.isExist() "+e.getMessage());
}
return x;
}
/**
* 关闭FTP连接,关闭时候像服务器发送一条关闭命令
* @param client FTP客户端
* @return 关闭成功,或者链接已断开,或者链接为null时候返回true,通过两次关闭都失败时候返回false
*/
public static boolean closeConn(FTPClient client) {
if (client == null) return true;
if (client.isConnected()) {
try {
client.disconnect(true);
return true;
} catch (Exception e) {
try {
client.disconnect(false);
} catch (Exception e1) {}
}
}
return true;
}
//###---------------------------------------------------------------------
/**
* 格式化文件路径,将其中不规范的分隔转换为标准的分隔符,并且去掉末尾的文件路径分隔符。
* 本方法操作系统自适应
* @param path 文件路径
* @return 格式化后的文件路径
*/
public static String formatPath4File(String path) {
String reg0 = "\\\\+";
String reg = "\\\\+|/+";
String temp = path.trim().replaceAll(reg0, "/");
temp = temp.replaceAll(reg, "/");
if (temp.length() > 1 && temp.endsWith("/")) {
temp = temp.substring(0, temp.length() - 1);
}
temp = temp.replace('/', File.separatorChar);
return temp;
}
/**
* 格式化文件路径,将其中不规范的分隔转换为标准的分隔符
* 并且去掉末尾的"/"符号(适用于FTP远程文件路径或者Web资源的相对路径)。
* @param path 文件路径
* @return 格式化后的文件路径
*/
public static String formatPath4FTP(String path) {
String reg0 = "\\\\+";
String reg = "\\\\+|/+";
String temp = path.trim().replaceAll(reg0, "/");
temp = temp.replaceAll(reg, "/");
if (temp.length() > 1 && temp.endsWith("/")) {
temp = temp.substring(0, temp.length() - 1);
}
return temp;
}
/**
* 获取FTP路径的父路径,但不对路径有效性做检查
* @param path FTP路径
* @return 父路径,如果没有父路径,则返回null
*/
public static String genParentPath4FTP(String path) {
String f = new File(path).getParent();
if (f == null) {
return null;
}else {
return formatPath4FTP(f);
}
}
//###---------------------------------------------------------------------
//获取指定目录下的文件
public static File[] getPathFiles(String path){
File file = new File(path);
if (file.isDirectory()) {
File[] files = file.listFiles();
return files;
}
return null;
}
//删除指定目录下的临时文件
public static void deleteFiles(String path){
File file = new File(path);
if (file.isDirectory()) {
File[] files = file.listFiles();
for (int i = 0; i < files.length; i++) {
//String name = files[i].getName();
//if(name.trim().toLowerCase().endsWith("_ftp.tmp")) {
//System.out.println(name + "\t");
//}
//清空临时文件 _ftp.tmp
files[i].delete();
}
}
}
//截取文件名称
public static String getFileName(String fileName){
String fs = "F"+System.currentTimeMillis();
try {
if(fileName != null && !"".equals(fileName)){
if(fileName.lastIndexOf("/") > 0 && fileName.lastIndexOf(".") > 0){
fs = fileName.substring(fileName.lastIndexOf("/")+1);
}else{
fs = fileName;
}
if(fs.length() >50){
fs = fs.substring(0,50);
}
return fs;
}else{
return fs;
}
} catch (Exception e) {
System.out.println("###[Error] FtpTools.getFileName()"+e.getMessage());
}
return fs;
}
public static void main(String args[]) throws Exception {
FTPClient f41 = FTPToolkit.ftpConn("129.2.8.31", 21, "root", "root31");
FTPClient f42 = FTPToolkit.ftpConn("129.2.8.32", 21, "root", "root32");
//InputStream in = FTPToolkit.fetchInputStream(f41,"/home/photofile/wenzi-003.jpg");
//FTPToolkit.uploadInputStream(f41,"/home/photofile/aa.jpg",in);
//FTPToolkit.upload(f41, "E:\\demo\\av1.jpg", "/home/photofile/10/20");
//FTPToolkit.download(f41, "/home/photofile/wenzi-100.jpg", "E:\\");
//FTPToolkit.getlistFiles(f41,"/home/photofile");
//文件流对流读取操作
List<String> ls = null;//FTPToolkit.getlistFileNames(f41,"/home/photofile");
if(ls != null && ls.size() >0){
InputStream in=null;
for(int i =0;i<ls.size();i++){
//in = FTPToolkit.fetchInputStream(f41,ls.get(i));
if(in != null && in.available() > 0){
FTPToolkit.uploadInputStream(f42,"/home/wasadmin/photoSysTempFTP/"+getFileName(ls.get(i)),in);
f41.deleteFile(ls.get(i));
}
}
//清空项目下的临时文件
FTPToolkit.deleteFiles("ftptmp");
}
FTPToolkit.closeConn(f41);
FTPToolkit.closeConn(f42);
}
}
/**
* FTP监听器
*/
class MyFtpListener implements FTPDataTransferListener {
private String tag;
private MyFtpListener(String tags) {
this.tag = tags;
}
public static MyFtpListener instance(String tags) {
return new MyFtpListener(tags);
}
public void started() {
//System.out.println(tag+ ":FTP启动。。。。。。");
}
public void transferred(int length) {
//System.out.println(tag+ ":FTP传输["+length+"]。。。。。。");
}
public void completed() {
//System.out.println(tag+ ":FTP完成。。。。。。");
}
public void aborted() {
//System.out.println(tag+ ":FTP中止。。。。。。");
}
public void failed() {
//System.out.println(tag+ ":FTP挂掉。。。。。。");
}
}