Java FTPClient代码(二)支持多线程与连接复用

2024-01-05 18:38

本文主要是介绍Java FTPClient代码(二)支持多线程与连接复用,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

该版本主要增加了以下特性:
1、对多线程并发的支持
2、连接复用


import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.net.SocketException;import org.apache.commons.net.ftp.FTPClient;
import org.apache.commons.net.ftp.FTPReply;
import org.apache.log4j.Logger;/*** FTP客户端* * @author summersun_ym* @version $Id: FTPClientTemplate.java 2010-11-22 上午12:54:47 $*/
public class FTPClientTemplate {//---------------------------------------------------------------------// Instance data//---------------------------------------------------------------------/** logger */protected final Logger         log                  = Logger.getLogger(getClass());private ThreadLocal<FTPClient> ftpClientThreadLocal = new ThreadLocal<FTPClient>();private String                 host;private int                    port;private String                 username;private String                 password;private boolean                binaryTransfer       = true;private boolean                passiveMode          = true;private String                 encoding             = "UTF-8";private int                    clientTimeout        = 1000 * 30;public String getHost() {return host;}public void setHost(String host) {this.host = host;}public int getPort() {return port;}public void setPort(int port) {this.port = port;}public String getUsername() {return username;}public void setUsername(String username) {this.username = username;}public String getPassword() {return password;}public void setPassword(String password) {this.password = password;}public boolean isBinaryTransfer() {return binaryTransfer;}public void setBinaryTransfer(boolean binaryTransfer) {this.binaryTransfer = binaryTransfer;}public boolean isPassiveMode() {return passiveMode;}public void setPassiveMode(boolean passiveMode) {this.passiveMode = passiveMode;}public String getEncoding() {return encoding;}public void setEncoding(String encoding) {this.encoding = encoding;}public int getClientTimeout() {return clientTimeout;}public void setClientTimeout(int clientTimeout) {this.clientTimeout = clientTimeout;}//---------------------------------------------------------------------// private method//---------------------------------------------------------------------/*** 返回一个FTPClient实例* * @throws FTPClientException*/private FTPClient getFTPClient() throws FTPClientException {if (ftpClientThreadLocal.get() != null && ftpClientThreadLocal.get().isConnected()) {return ftpClientThreadLocal.get();} else {FTPClient ftpClient = new FTPClient(); //构造一个FtpClient实例ftpClient.setControlEncoding(encoding); //设置字符集connect(ftpClient); //连接到ftp服务器//设置为passive模式if (passiveMode) {ftpClient.enterLocalPassiveMode();}setFileType(ftpClient); //设置文件传输类型try {ftpClient.setSoTimeout(clientTimeout);} catch (SocketException e) {throw new FTPClientException("Set timeout error.", e);}ftpClientThreadLocal.set(ftpClient);return ftpClient;}}/*** 设置文件传输类型* * @throws FTPClientException* @throws IOException*/private void setFileType(FTPClient ftpClient) throws FTPClientException {try {if (binaryTransfer) {ftpClient.setFileType(FTPClient.BINARY_FILE_TYPE);} else {ftpClient.setFileType(FTPClient.ASCII_FILE_TYPE);}} catch (IOException e) {throw new FTPClientException("Could not to set file type.", e);}}/*** 连接到ftp服务器* * @param ftpClient* @return 连接成功返回true,否则返回false* @throws FTPClientException*/private boolean connect(FTPClient ftpClient) throws FTPClientException {try {ftpClient.connect(host, port);// 连接后检测返回码来校验连接是否成功int reply = ftpClient.getReplyCode();if (FTPReply.isPositiveCompletion(reply)) {//登陆到ftp服务器if (ftpClient.login(username, password)) {setFileType(ftpClient);return true;}} else {ftpClient.disconnect();throw new FTPClientException("FTP server refused connection.");}} catch (IOException e) {if (ftpClient.isConnected()) {try {ftpClient.disconnect(); //断开连接} catch (IOException e1) {throw new FTPClientException("Could not disconnect from server.", e1);}}throw new FTPClientException("Could not connect to server.", e);}return false;}//---------------------------------------------------------------------// public method//---------------------------------------------------------------------/*** 断开ftp连接* * @throws FTPClientException*/public void disconnect() throws FTPClientException {try {FTPClient ftpClient = getFTPClient();ftpClient.logout();if (ftpClient.isConnected()) {ftpClient.disconnect();ftpClient = null;}} catch (IOException e) {throw new FTPClientException("Could not disconnect from server.", e);}}public boolean mkdir(String pathname) throws FTPClientException {return mkdir(pathname, null);}/*** 在ftp服务器端创建目录(不支持一次创建多级目录)* * 该方法执行完后将自动关闭当前连接* * @param pathname* @return* @throws FTPClientException*/public boolean mkdir(String pathname, String workingDirectory) throws FTPClientException {return mkdir(pathname, workingDirectory, true);}/*** 在ftp服务器端创建目录(不支持一次创建多级目录)* * @param pathname* @param autoClose 是否自动关闭当前连接* @return* @throws FTPClientException*/public boolean mkdir(String pathname, String workingDirectory, boolean autoClose) throws FTPClientException {try {getFTPClient().changeWorkingDirectory(workingDirectory);return getFTPClient().makeDirectory(pathname);} catch (IOException e) {throw new FTPClientException("Could not mkdir.", e);} finally {if (autoClose) {disconnect(); //断开连接}}}/*** 上传一个本地文件到远程指定文件* * @param remoteAbsoluteFile 远程文件名(包括完整路径)* @param localAbsoluteFile 本地文件名(包括完整路径)* @return 成功时,返回true,失败返回false* @throws FTPClientException*/public boolean put(String remoteAbsoluteFile, String localAbsoluteFile) throws FTPClientException {return put(remoteAbsoluteFile, localAbsoluteFile, true);}/*** 上传一个本地文件到远程指定文件* * @param remoteAbsoluteFile 远程文件名(包括完整路径)* @param localAbsoluteFile 本地文件名(包括完整路径)* @param autoClose 是否自动关闭当前连接* @return 成功时,返回true,失败返回false* @throws FTPClientException*/public boolean put(String remoteAbsoluteFile, String localAbsoluteFile, boolean autoClose) throws FTPClientException {InputStream input = null;try {// 处理传输input = new FileInputStream(localAbsoluteFile);getFTPClient().storeFile(remoteAbsoluteFile, input);log.debug("put " + localAbsoluteFile);return true;} catch (FileNotFoundException e) {throw new FTPClientException("local file not found.", e);} catch (IOException e) {throw new FTPClientException("Could not put file to server.", e);} finally {try {if (input != null) {input.close();}} catch (Exception e) {throw new FTPClientException("Couldn't close FileInputStream.", e);}if (autoClose) {disconnect(); //断开连接}}}/*** 下载一个远程文件到本地的指定文件* * @param remoteAbsoluteFile 远程文件名(包括完整路径)* @param localAbsoluteFile 本地文件名(包括完整路径)* @return 成功时,返回true,失败返回false* @throws FTPClientException*/public boolean get(String remoteAbsoluteFile, String localAbsoluteFile) throws FTPClientException {return get(remoteAbsoluteFile, localAbsoluteFile, true);}/*** 下载一个远程文件到本地的指定文件* * @param remoteAbsoluteFile 远程文件名(包括完整路径)* @param localAbsoluteFile 本地文件名(包括完整路径)* @param autoClose 是否自动关闭当前连接* * @return 成功时,返回true,失败返回false* @throws FTPClientException*/public boolean get(String remoteAbsoluteFile, String localAbsoluteFile, boolean autoClose) throws FTPClientException {OutputStream output = null;try {output = new FileOutputStream(localAbsoluteFile);return get(remoteAbsoluteFile, output, autoClose);} catch (FileNotFoundException e) {throw new FTPClientException("local file not found.", e);} finally {try {if (output != null) {output.close();}} catch (IOException e) {throw new FTPClientException("Couldn't close FileOutputStream.", e);}}}/*** 下载一个远程文件到指定的流 处理完后记得关闭流* * @param remoteAbsoluteFile* @param output* @return* @throws FTPClientException*/public boolean get(String remoteAbsoluteFile, OutputStream output) throws FTPClientException {return get(remoteAbsoluteFile, output, true);}/*** 下载一个远程文件到指定的流 处理完后记得关闭流* * @param remoteAbsoluteFile* @param output* @param delFile* @return* @throws FTPClientException*/public boolean get(String remoteAbsoluteFile, OutputStream output, boolean autoClose) throws FTPClientException {try {FTPClient ftpClient = getFTPClient();// 处理传输return ftpClient.retrieveFile(remoteAbsoluteFile, output);} catch (IOException e) {throw new FTPClientException("Couldn't get file from server.", e);} finally {if (autoClose) {disconnect(); //关闭链接}}}/*** 从ftp服务器上删除一个文件* 该方法将自动关闭当前连接* * @param delFile* @return* @throws FTPClientException*/public boolean delete(String delFile) throws FTPClientException {return delete(delFile, true);}/*** 从ftp服务器上删除一个文件* * @param delFile* @param autoClose 是否自动关闭当前连接* * @return* @throws FTPClientException*/public boolean delete(String delFile, boolean autoClose) throws FTPClientException {try {getFTPClient().deleteFile(delFile);return true;} catch (IOException e) {throw new FTPClientException("Couldn't delete file from server.", e);} finally {if (autoClose) {disconnect(); //关闭链接}}}/*** 批量删除* 该方法将自动关闭当前连接* * @param delFiles* @return* @throws FTPClientException*/public boolean delete(String[] delFiles) throws FTPClientException {return delete(delFiles, true);}/*** 批量删除* * @param delFiles* @param autoClose 是否自动关闭当前连接* * @return* @throws FTPClientException*/public boolean delete(String[] delFiles, boolean autoClose) throws FTPClientException {try {FTPClient ftpClient = getFTPClient();for (String s : delFiles) {ftpClient.deleteFile(s);}return true;} catch (IOException e) {throw new FTPClientException("Couldn't delete file from server.", e);} finally {if (autoClose) {disconnect(); //关闭链接}}}/*** 列出远程默认目录下所有的文件* * @return 远程默认目录下所有文件名的列表,目录不存在或者目录下没有文件时返回0长度的数组* @throws FTPClientException*/public String[] listNames() throws FTPClientException {return listNames(null, true);}public String[] listNames(boolean autoClose) throws FTPClientException {return listNames(null, autoClose);}/*** 列出远程目录下所有的文件* * @param remotePath 远程目录名* @param autoClose 是否自动关闭当前连接* * @return 远程目录下所有文件名的列表,目录不存在或者目录下没有文件时返回0长度的数组* @throws FTPClientException*/public String[] listNames(String remotePath, boolean autoClose) throws FTPClientException {try {String[] listNames = getFTPClient().listNames(remotePath);return listNames;} catch (IOException e) {throw new FTPClientException("列出远程目录下所有的文件时出现异常", e);} finally {if (autoClose) {disconnect(); //关闭链接}}}public static void main(String[] args) throws FTPClientException, InterruptedException {FTPClientTemplate ftp = new FTPClientTemplate();ftp.setHost("localhost");ftp.setPort(2121);ftp.setUsername("admin");ftp.setPassword("admin");ftp.setBinaryTransfer(false);ftp.setPassiveMode(false);ftp.setEncoding("utf-8");//boolean ret = ftp.put("/group/tbdev/query/user-upload/12345678910.txt", "D:/099_temp/query/12345.txt");//System.out.println(ret);ftp.mkdir("asd", "user-upload");//ftp.disconnect();//ftp.mkdir("user-upload1");//ftp.disconnect();//String[] aa = {"/group/tbdev/query/user-upload/123.txt", "/group/tbdev/query/user-upload/SMTrace.txt"};//ftp.delete(aa);}
}复用

这篇关于Java FTPClient代码(二)支持多线程与连接复用的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/573778

相关文章

Spring事务传播机制最佳实践

《Spring事务传播机制最佳实践》Spring的事务传播机制为我们提供了优雅的解决方案,本文将带您深入理解这一机制,掌握不同场景下的最佳实践,感兴趣的朋友一起看看吧... 目录1. 什么是事务传播行为2. Spring支持的七种事务传播行为2.1 REQUIRED(默认)2.2 SUPPORTS2

怎样通过分析GC日志来定位Java进程的内存问题

《怎样通过分析GC日志来定位Java进程的内存问题》:本文主要介绍怎样通过分析GC日志来定位Java进程的内存问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、GC 日志基础配置1. 启用详细 GC 日志2. 不同收集器的日志格式二、关键指标与分析维度1.

Java进程异常故障定位及排查过程

《Java进程异常故障定位及排查过程》:本文主要介绍Java进程异常故障定位及排查过程,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、故障发现与初步判断1. 监控系统告警2. 日志初步分析二、核心排查工具与步骤1. 进程状态检查2. CPU 飙升问题3. 内存

java中新生代和老生代的关系说明

《java中新生代和老生代的关系说明》:本文主要介绍java中新生代和老生代的关系说明,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、内存区域划分新生代老年代二、对象生命周期与晋升流程三、新生代与老年代的协作机制1. 跨代引用处理2. 动态年龄判定3. 空间分

Java设计模式---迭代器模式(Iterator)解读

《Java设计模式---迭代器模式(Iterator)解读》:本文主要介绍Java设计模式---迭代器模式(Iterator),具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,... 目录1、迭代器(Iterator)1.1、结构1.2、常用方法1.3、本质1、解耦集合与遍历逻辑2、统一

Java内存分配与JVM参数详解(推荐)

《Java内存分配与JVM参数详解(推荐)》本文详解JVM内存结构与参数调整,涵盖堆分代、元空间、GC选择及优化策略,帮助开发者提升性能、避免内存泄漏,本文给大家介绍Java内存分配与JVM参数详解,... 目录引言JVM内存结构JVM参数概述堆内存分配年轻代与老年代调整堆内存大小调整年轻代与老年代比例元空

深度解析Java DTO(最新推荐)

《深度解析JavaDTO(最新推荐)》DTO(DataTransferObject)是一种用于在不同层(如Controller层、Service层)之间传输数据的对象设计模式,其核心目的是封装数据,... 目录一、什么是DTO?DTO的核心特点:二、为什么需要DTO?(对比Entity)三、实际应用场景解析

Java 线程安全与 volatile与单例模式问题及解决方案

《Java线程安全与volatile与单例模式问题及解决方案》文章主要讲解线程安全问题的五个成因(调度随机、变量修改、非原子操作、内存可见性、指令重排序)及解决方案,强调使用volatile关键字... 目录什么是线程安全线程安全问题的产生与解决方案线程的调度是随机的多个线程对同一个变量进行修改线程的修改操

从原理到实战深入理解Java 断言assert

《从原理到实战深入理解Java断言assert》本文深入解析Java断言机制,涵盖语法、工作原理、启用方式及与异常的区别,推荐用于开发阶段的条件检查与状态验证,并强调生产环境应使用参数验证工具类替代... 目录深入理解 Java 断言(assert):从原理到实战引言:为什么需要断言?一、断言基础1.1 语

深度解析Java项目中包和包之间的联系

《深度解析Java项目中包和包之间的联系》文章浏览阅读850次,点赞13次,收藏8次。本文详细介绍了Java分层架构中的几个关键包:DTO、Controller、Service和Mapper。_jav... 目录前言一、各大包1.DTO1.1、DTO的核心用途1.2. DTO与实体类(Entity)的区别1