开发问题解决---FTP线程池使用报错connection reset by peer, socket write error

背景

内部重构的一个项目中,同事通过apache的GenericObjectPool构建一个小型的线程池维护FTP服务器的连接,但是在实际使用中偶尔会出去报错,错误为:connection reset by peer, socket write error

分析

很明显,这是一个socket连接重置的问题。在服务器和客户端交互的模型中,两方都维护着发送端和接收端,主动发送(output)与被动接收(input),当服务器接收端已经关闭后,而客户端却还在发送数据的时候,服务器就会向客户端发送(连接已经重置,socket错误)的消息。ftp导致报错的主要原因是连接超时。

解决方法

在连接池的的borrowObject()方法中,默认不会校验去除的对象,但是可以设置取出的对象是否校验通过,通过setTestOnBorrow(true)设置获取对象时进行校验,具体代码在borrowObject(long borrowMaxWaitMillis)
在这里插入图片描述
这样保证每次从线程池中获取的对象都可用。

  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
如果在多线程环境下使用 Jedis 时出现 Connection reset by peer: socket write error 的错误,可能是因为 Jedis 不是线程安全的。可以通过以下两种方式解决这个问题: 1. 使用线程安全的 JedisPool JedisPool 可以创建多个 Jedis 实例,并提供线程安全的连接池管理。可以在每个线程中获取一个 Jedis 实例来使用使用完毕后归还到连接池中,从而避免线程之间的竞争和冲突。 ```java import redis.clients.jedis.Jedis; import redis.clients.jedis.JedisPool; import redis.clients.jedis.JedisPoolConfig; public class JedisDemo { private static JedisPool jedisPool; static { JedisPoolConfig config = new JedisPoolConfig(); config.setMaxTotal(10); config.setMaxIdle(5); config.setTestOnBorrow(true); jedisPool = new JedisPool(config, "localhost", 6379); } public static void main(String[] args) { Jedis jedis = jedisPool.getResource(); try { jedis.set("name", "JedisDemo"); String value = jedis.get("name"); System.out.println("name = " + value); } finally { if (jedis != null) { jedis.close(); } } jedisPool.close(); } } ``` 2. 在每个线程中创建新的 Jedis 实例 在每个线程中创建新的 Jedis 实例,使用完毕后立即关闭连接。这种方式会频繁地创建和关闭连接,不够高效,但是适用于并发量不高的场景。 ```java import redis.clients.jedis.Jedis; public class JedisDemo { public static void main(String[] args) { Thread thread1 = new Thread(() -> { Jedis jedis = new Jedis("localhost", 6379); try { jedis.set("name", "JedisDemo1"); String value = jedis.get("name"); System.out.println("name = " + value); } finally { if (jedis != null) { jedis.close(); } } }); thread1.start(); Thread thread2 = new Thread(() -> { Jedis jedis = new Jedis("localhost", 6379); try { jedis.set("name", "JedisDemo2"); String value = jedis.get("name"); System.out.println("name = " + value); } finally { if (jedis != null) { jedis.close(); } } }); thread2.start(); } } ``` 这个示例在两个线程中分别创建新的 Jedis 实例,设置一个键值对,获取键值对的值并打印输出,最后关闭连接。由于每个线程都使用自己的 Jedis 实例,不会出现线程之间的竞争和冲突。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值