解决Redis连接无法正常释放的问题
一、问题描述
前些天用多线程执行操作测试验证vanyar-redis连接池,应用是刚重启的状态,执行操作是,开启10个线程同时执行10000次操作。
如下:
执行操作完毕后发现控制台输出9个下面错误信息:
该错误大致意思是说:不能将redis连接放回池内,放回连接池的对象是无效的对象。在网上查了很多同类错误,都说是进行了两次returnResource释放连接资源造成的,因为第一次return成功以后,第二次return就会报上面这个错误。但是显然,我翻遍了代码并没有两次调用returnResource。
查看redis服务端的连接数详细信息如下,前9个连接,idle=453,空闲了453秒了,依然没有释放,而连接池设置的是空闲60秒就会被释放,明显发生异常了。
初步怀疑是多线程执行redis操作,初始化redis连接池有问题。于是重启应用,先执行单线程redis操作,再执行多线程redis操作,没有发生上面的问题。redis服务端连接均能正常释放。由此得出结论,当线程池在未初始化的时候,由于多线程同时执行redis连接池初始化工作引起的问题。
看代码(RedisJedisPool未优化之前):当10个线程同时请求redis连接资源时,10个线程都发现连接池为空(因为创建连接池相比创建线程比较耗时),这时10个线程都各自初始化成功一个连接池,并从中取得redis连接,并执行了redis操作。执行完毕,returnResource的时候,由于此时pool变量的引用是最后一个线程初始化的连接池,前面9个线程获得的redis连接并不属于最后一个连接池的资源,所以抛错:IllegalStateException: Invalidated object not currently part of this pool
二、报错原因分析
线程1 : 创建redis连接池1 : 获得redis连接1
线程2 : 创建redis连接池2 : 获得redis连接2
线程3 : 创建redis连接池3 : 获得redis连接3
……
线程8 : 创建redis连接池8 : 获得redis连接8
线程9 : 创建redis连接池9 : 获得redis连接9
线程10 : 创建redis连接池10 : 获得redis连接10
全局变量pool引用 指向 redis连接池10
当线程1-9 把redis连接1-9 归还给pool-redis连接池10
reds连接池10自然就报错,说:
IllegalStateException: Invalidated object not currently part of this pool
三、解决办法
由于创建线程池,连接池等工作都是相对比较耗时的,所以我们一般放在应用启动的时候就初始化,把连接池的初始化工作交给Spring容器管理,同时把初始化连接池和获取连接两个操作实现方法分离,对初始化连接池的方法加上同步锁机制,并且二次判断是否为空,就算多线程情况下,在二次判断是否为空的时候,pool已经不为空了,直接返回。现在多线程安全的问题就得以解决。
附上,解决前后对比图:
补充知识:java spring框架中方法级redis的连接自动获取和释放实现
java中使用redis总是需要处理redis连接的获取,释放等操作,每次使用都会使代码变的特别丑陋,模仿spring中aop的实现,用动态代理写一个 连接自动获取和释放的工具
主要思路
JedisManageSupport 抽象类 类似于 aop的切入点,所有继承了该类(一般都是service层)的类,可以使用提供的获取redis的方法获取redis,并且不需要释放
JedisBeanPostProcessor 继承BeanPostProcessor ,会在bean初始化时执行自己定义的逻辑:
如果A类继承了 JedisManageSupport ,就会获取redis连接并且放到JedisManageSupport 的成员变量里,A类的实例(其实是cglib动态代理生成的
A类的子类的实例)就可以使用该redis连接 进行相关操作了
代理类的实例见源码
源码如下
public class JedisBeanPostProcessor implements BeanPostProcessor { @Autowired ShardedJedisPool shardedJedisPool; static final Logger logger = Logger.getLogger(JedisBeanPostProcessor.class); @Override public Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException { if (bean instanceof JedisManageSupport) { Enhancer enhancer = new Enhancer(); enhancer.setSuperclass(bean.getClass()); enhancer.setCallback(new JedisInterceptor(shardedJedisPool, bean)); Object targetBean = enhancer.create(); return targetBean; else { return bean; @Override public Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException { return bean; class JedisInterceptor implements MethodInterceptor { static final Logger logger = Logger.getLogger(JedisInterceptor.class); ShardedJedisPool pool; Object src; public JedisInterceptor(ShardedJedisPool pool, Object src) { this.pool = pool; this.src = src; @Override public Object intercept(Object target, Method method, Object[] arguments, MethodProxy methodProxy) throws Throwable { Object result = null; if (target instanceof JedisManageSupport) { if (this.isDeclaredMethod(target, method)) { ShardedJedis jedis = null; try { JedisManageSupport support = (JedisManageSupport) src; jedis = pool.getResource(); support.setShardedJedis(jedis); // logger.debug("调用之前注入jedis对象,method:" + method); * 下面代码可以使用 method.invoke(src,arguments)。 不能使用 * methodProxy.invokeSuper(target,arguments); * 因为A类中用Autowired注入的属性,生成代理的子类B后,因为子类B是新建的类。从父类继承的属性没有被初始化, * 使用methodProxy.invokeSuper()执行是,会报空指针异常. result = methodProxy.invoke(src, arguments); support.setShardedJedis(null); catch (Exception e) { pool.returnBrokenResource(jedis); e.printStackTrace(); finally { if (jedis != null) { pool.returnResource(jedis); // logger.debug("调用之后归还jedis对象,method:" + method); else { result = methodProxy.invoke(src, arguments); else { throw new Exception("使用该代理必须继承JedisManageSupport"); return result; * 是否是target类本身定义的非私有的方法,还是继承的父类 * @return true是target自己类的并且不是私有的的, private boolean isDeclaredMethod(Object target, Method arg1) { Method temp = null; try { temp = target.getClass().getDeclaredMethod(arg1.getName(), arg1.getParameterTypes()); catch (SecurityException e) { e.printStackTrace(); catch (NoSuchMethodException e) { e.printStackTrace(); * 不为null,并且是非私有的,返回true if (temp != null) { return true; else { return false; public abstract class JedisManageSupport { ThreadLocal ShardedJedis jedisHolder = new ThreadLocal ShardedJedis public final ShardedJedis getShardedJedis() { return jedisHolder.get(); public final void setShardedJedis(ShardedJedis jedis) { jedisHolder.set(jedis); * 如果某个键不同单位之间也不会重复,可以使用这个方法生成redis的键 public final byte[] assemKey(String baseKey) { Assert.isTrue(StringUtils.isNotBlank(baseKey), "参数不能为空"); return baseKey.getBytes(); * 根据tableName+prefix 构造唯一key与assemKey(String baseKey, String tableName) * 规则一致 public final byte[] assemKeyByPrefix(String tableName, String baseKey) { Assert.isTrue(StringUtils.isNotBlank(baseKey), "参数不能为空"); Assert.isTrue(StringUtils.isNotBlank(tableName), "参数不能为空"); UnitInfo unit = WebService.getUnitInfo(); Assert.isTrue(unit != null, "单位信息获取不到"); return (tableName + "-" + unit.getPrefix() + "-" + baseKey).getBytes(); * 不同前缀的表中可能有相同的键,同一个表中也可能是有重复的baseKey时,用这个生成redis的key 比如 用户信息表的 * username字段,不同的用户信息表允许重复的username,mooc_t_userinfo * 也允许有相同的账号,所以生成redis的key时,需要用到单位的mooc_school 放入redis中 public final byte[] assemKeyByFid(String tableName, String baseKey) { UnitInfo unit = WebService.getUnitInfo(); Assert.isTrue(unit != null, "单位信息获取不到"); return (tableName + "-" + unit.getMoocSchool() + "-" + baseKey).getBytes();
以上这篇解决Redis连接无法正常释放的问题就是小编分享给大家的全部内容了,希望能给大家一个参考,也希望大家多多支持。
我想要获取技术服务或软件
服务范围:MySQL、ORACLE、SQLSERVER、MongoDB、PostgreSQL 、程序问题
服务方式:远程服务、电话支持、现场服务,沟通指定方式服务
技术标签:数据恢复、安装配置、数据迁移、集群容灾、异常处理、其它问题
本站部分文章参考或来源于网络,如有侵权请联系站长。
数据库远程运维 解决Redis连接无法正常释放的问题
相关文章
- Redis中文手册 畅游Redis世界(redis中文手册)
- 行连接Redis数据库:使用命令行管理(连接redis数据库命令)
- 深入理解Redis的数据存储类型(redis数据存储类型)
- 红色之火:链接到 Redis 数据库(redis 数据库连接)
- 深入浅出理解Redis的多线程特性(怎么理解redis多线程)
- 微服务系统的Redis连接优化之路(微服务redis连接数)
- 使用Redis解决并发多大的问题(并发多大要用到redis)
- 解放Redis服务解决Redis穿透问题(解决redis穿透问题)
- 虚拟主机上的Redis开启智能数据管理(虚拟主机使用redis)
- 深入体会编译Redis源文件(编译redis源文件)
- 探究系统中是否安装了Redis(查询有没有安装redis)
- 精准操作服务器如何连接Redis(服务器怎么连redis)
- 无法解决本地Redis连接问题(无法连接本地redis)
- SSB强势加入Redis,全新改写架构蓝图(ssb整合redis)
- Redis 快速列表序列化技术研究(列表序列化到redis)
- 步步为营安装完Redis,如何与之连接(安装好redis怎么连接)
- 玩转Redis实现多客户端连接(多客户端连接redis)
- 解放双手用Redis提高性能(redis高性能怎么解决)
- Redis 高压下的连接困境(redis高压后连接不上)
- 接Redis集群实现连接的最佳实践(redis集群怎么连)
- 解决Redis集群遇到的挑战(redis集群存在的问题)
- Redis集群高效使用Set命令(redis集群set命令)
- 探究Redis阻塞原理及实现机制(redis 阻塞原理)
- 策略Redis队列入库优化时效性策略实现(redis队列入库时效)
- 探讨Redis队列的冲突问题(redis队列会冲突吗)
- 解决Redis链表过大问题的新思路(redis 链表过大)
- 简单搭建Redis实现消息队列功能(redis连接消息队列)
- Redis连接池断线重连保持连接的持久性(redis连接池断线重连)
- IDEA中快速搭建Redis连接(redis连接idea)
- 利用Redis软件实现跨平台的连接(redis软件连接)
- Redis架设安全防线,体验快捷登录(redis 设置登录密码)
- Redis自动启动功能的设置(redis设置成自动启动)
- Redis过期多线程处理加速平衡(redis过期 多线程)
- 解决Redis缓存中的挑战(redis 缓存问题)