common-pool2配置ftpClient对象连接池

依赖

<dependency>
     <groupId>org.apache.commons</groupId>
     <artifactId>commons-pool2</artifactId>
</dependency>
<dependency>
     <groupId>commons-net</groupId>
     <artifactId>commons-net</artifactId>
     <version>3.5</version>
</dependency>

yml

ftp:
#  enableConfig: false表示配置都使用默认配置
#  enableConfig: true
  host: 192.168.52.44
  port: 21
  username: ftpuser
  password: 123456
  basePath: /home/ftp
  pool:
    #池对象耗尽之后是否阻塞
    blockWhenExhausted: true
    #最大数
    maxTotal: 8
    #最大空闲
    maxIdle: 8
    #最小空闲
    minIdle: 0
    #最大等待时间
    maxWait: 1000
    #空闲验证
    testWhileIdle: true

FtpClientPoolFactory

@Log4j2
@Configuration
public class FtpClientPoolFactory implements  PooledObjectFactory<FTPClient> {
    @Value("${ftp.host}")
    private String ftpHost;
    @Value("${ftp.port}")
    private Integer ftpPort;
    @Value("${ftp.username}")
    private String ftpUsername;
    @Value("${ftp.password}")
    private String ftpPassword;
    @Override
    public void activateObject(PooledObject<FTPClient> pooledObject) throws Exception {

    }

    @Override
    public void destroyObject(PooledObject<FTPClient> pooledObject){
        FTPClient object = pooledObject.getObject();
        if(object!=null && object.isConnected()){
            try {
                object.logout();
            } catch (IOException e) {
            }finally {
                try {
                    object.disconnect();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
        }
    }

    @Override
    public PooledObject<FTPClient> makeObject() throws Exception {
        FTPClient ftpClient = new FTPClient();
        ftpClient.connect(ftpHost, ftpPort);
        ftpClient.login(ftpUsername, ftpPassword);
        if (!FTPReply.isPositiveCompletion(ftpClient.getReplyCode())) {
            ftpClient.disconnect();
            throw new RuntimeException("未连接到FTP,用户名或密码错误。");
        }
        return new DefaultPooledObject<>(ftpClient);
    }

    @Override
    public void passivateObject(PooledObject<FTPClient> pooledObject) throws Exception {

    }

    @Override
    public boolean validateObject(PooledObject<FTPClient> pooledObject) {
        FTPClient object = pooledObject.getObject();
        try {
            return object!=null && object.isConnected() && object.sendNoOp();
        } catch (IOException e) {
            e.printStackTrace();
        }
        return false;
    }
}

FtpClientPool

@Component
public class FtpClientPool {
    private GenericObjectPool<FTPClient> pool;
    @Value("${ftp.enableConfig}")
    private Boolean enableConfig;
    @Value("${ftp.pool.blockWhenExhausted}")
    private Boolean blockWhenExhausted;
    @Value("${ftp.pool.maxTotal}")
    private Integer maxTotal;
    @Value("${ftp.pool.maxIdle}")
    private Integer maxIdle;
    @Value("${ftp.pool.minIdle}")
    private Integer minIdle;
    @Value("${ftp.pool.maxWait}")
    private Integer maxWait;
    @Value("${ftp.pool.testWhileIdle}")
    private Boolean testWhileIdle;

    public FtpClientPool(PooledObjectFactory<FTPClient> ftpClientPoolFactory) {
        if(enableConfig==null || !enableConfig){
            pool= new GenericObjectPool<>(ftpClientPoolFactory);
            return;
        }
        GenericObjectPoolConfig<FTPClient> poolConfig=new GenericObjectPoolConfig<>();
        poolConfig.setBlockWhenExhausted(blockWhenExhausted);
        poolConfig.setMaxWaitMillis(maxWait);
        poolConfig.setMinIdle(minIdle);
        poolConfig.setMaxIdle(maxIdle);
        poolConfig.setMaxTotal(maxTotal);
        poolConfig.setTestOnBorrow(false);
        poolConfig.setTestOnReturn(false);
        poolConfig.setTestOnCreate(false);
        poolConfig.setTestWhileIdle(testWhileIdle);
        pool=new GenericObjectPool<>(ftpClientPoolFactory,poolConfig);
    }
    public FTPClient borrowObject(){
        try {
            return pool.borrowObject();
        } catch (Exception e) {
            e.printStackTrace();
        }
        return null;
    }

    public void returnObject(FTPClient obj) {
        pool.returnObject(obj);
    }
}

使用

    @Autowired
    private FtpClientPool ftpClientPool;
    @GetMapping("/upload")
    public void upload(){
        FTPClient ftpClient = ftpClientPool.borrowObject();
        //逻辑
        ftpClientPool.returnObject(ftpClient);
    }
经过几天的琢磨,去看了csdn上一位大牛的数据库的连接池实现方案,从中感悟很多,感谢这位大神。 让我才能有信心去坚持下去。也不知道写的好不好··不好的话,大家指出。但是我是努力去做了,这一个过程,很享受,大家互相学习吧~ 其实ftp连接池跟数据库连接池的原理是差不多的,不同的是ftp连接池有个连接时间的限制,如果你没设置的话,它的默认连接服务器的时间是0,所以我们要合理的设置它的服务器的时间,ftp.setConnectTimeout(5000);在这里设置了它的时间是5s。 写ftp连接池的目的就是合理的利用资源,本文的目的是在初始的时候,创建10个Ftp连接,放到一个队列中去,当多个用户同时去下载ftp上的文件的时候,就会从队列中取,若当前的队列中存在着空闲的连接,就获取该ftp的连接,并设置此连接为忙的状态,否则就在创建新的连接到连接池中去(有最大的连接池数的限制,不能超过这个连接数,超过的话,就会进入等待状态,直到其它连接释放连接),在执行下载操作的前对登录ftp时间进行判断。看是否超时,超时的话,就重新连接到ftp服务器,在这里我所做的操作就是,在开始创建ftp连接池的时候,记录下系统的当前时间,例如为:long beginTime=System.currentTimeMillis(),在取文件之前获得 当前系统的时间 long endTime=System.currentTimeMillis(),此时我们就可以获得系统登录ftp的时间time=endTime-beginTime,在此我们可以用time与ftp最大登录服务器时间(ftpPool.getConnection();)进行比较。 当然了,在操作完之后我们需要将所操作的连接池中的ftp设置为空闲状态。代码在文件中,为了测试,我本地自己创建了一个ftp服务器,创建ftp的方法,大家可以到网上查资料,我用的是Serv-U工具。傻瓜式的。所用到的jar包是commons-net2.0.
评论 4
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

可——叹——落叶飘零

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值