Scala/Java - Redis 连接检测与重试

本文涉及的产品
云数据库 Redis 版,社区版 2GB
推荐场景:
搭建游戏排行榜
简介: 项目实现中需要连接 redis,为了防止因网络抖动或其他原因造成的客户端连接失败,一般需要增加重试机制判断 client 是否连接成功,之前写了一版重连代码发现有 bug,借此机会看下代码 bug 以及如何更好的重连 redis。...

一.引言

项目实现中需要连接 redis,为了防止因网络抖动或其他原因造成的客户端连接失败,一般需要增加重试机制判断 client 是否连接成功,之前写了一版重连代码发现有 bug,借此机会看下代码 bug 以及如何更好的重连 redis。

二.错误代码

def getRedisClient(host: String, port: Int, timeout: Int = 0): Jedis = {
    var redisClient: Jedis = null
    var attempt = 3
    while (attempt > 0 && redisClient == null) {
      try {
        if (!timeout.equals(0)) {
          redisClient = new Jedis(host, port, timeout)
        } else {
          redisClient = new Jedis(host, port)
        }
      } catch {
        case e: Throwable => {
          e.printStackTrace()
          println(s"redis连接异常 $host 重试次数: $attempt")
          Thread.sleep(1000)
        }
      } finally {
        attempt -= 1
      }
    }
    redisClient
  }

image.gif

该代码为了实现报错重连机制,上线后发现存在下述问题 (短短几行代码这么多 bug o(╥﹏╥)o):

A.host 正确,不抛异常,client 不可用

出现该情况可能是网络抖动造成,不会进入 try catch 逻辑,但是后续使用该 client 会报错,因为redisClient = new Jedis() 方法没有返回值,所以只初始化无法判断该连接是否有效:

public Jedis(final String host, final int port, final int timeout) {
    super(host, port, timeout);
  }

image.gif

B.host 错误,抛异常,client 不可用

host 错误情况下,直接报错 java.net.UnknownHostException

C.client == null 判断问题

while 循环 client == null 作为判断条件,但只在第一次生效,因为第一次初始化后,即使是错误的 host 和 ip,client 也会变为 Jedis 对象而不再是 null,从而不会执行后续重试逻辑:

image.gif编辑

三.修改版

上面A,B,C三个问题总结下来就是两个问题:

(1) Jedis 客户端连接失败时需要抛出异常进入 catch 逻辑

(2) while 循环判断时需要新的判断条件

这个时候 ping 方法满足我们的需求,该方法返回 String,client 正常时返回 "PONG"(新的判断条件),client 无效时则抛出异常(进入catch逻辑),所以满足了上述要求:

public String ping() {
    checkIsInMulti();
    client.ping();
    return client.getStatusCodeReply();
  }

image.gif

这里将重试次数增加为 10 次,将判断条件修改为 ping 后返回的结果,同时保证 ping 后出问题可以打出异常日志方便后续问题排查:

def getRedisClient(host: String, port: Int, timeout: Int = 0): Jedis = {
  var redisClient: Jedis = null
  var attempt = 10
  var ping = ""
  while (attempt > 0 && ping != "PONG") {
    try {
      if (!timeout.equals(0) && attempt >= 10) {
        redisClient = new Jedis(host, port, timeout)
      } else {
        redisClient = new Jedis(host, port)
      } 
      ping = redisClient.ping()
    } catch {
      case e: Throwable => {
        e.printStackTrace()
        println(s"redis连接异常 $host 重试次数: $attempt")
        Thread.sleep(1000)
      }
    } finally {
      attempt -= 1
    }
  }
  println(s"初始化成功 ${host} ping: $ping")
  redisClient
}

image.gif

四.能不能再改进一点

redis 配置为主从结构,上述方法针对单台 redis 进行 10 次连接,一般情况下不会出问题,但是极端情况下还是会导致服务异常,所以可以在代码中增加另一台 host 的连接,从而保证 slave or master 单点故障的情况下服务仍高可用:

def getRedisClient(host: String, port: Int, timeout: Int = 0, other_host: String = ""): Jedis = {
  var redisClient: Jedis = null
  var attempt = 20
  var ping = ""
  while (attempt > 0 && ping != "PONG") {
    try {
      if (!timeout.equals(0) && attempt >= 10) {
        redisClient = new Jedis(host, port, timeout)
      } else if (attempt >= 10) {
        redisClient = new Jedis(host, port)
      } else if (!timeout.equals(0)) {
        redisClient = new Jedis(other_host, port, timeout)
      } else {
        redisClient = new Jedis(other_host, port)
      }
      ping = redisClient.ping()
    } catch {
      case e: Throwable => {
        e.printStackTrace()
        if (attempt >= 10) {
          println(s"redis连接异常 $host 重试次数: $attempt")
        } else {
          println(s"redis连接异常 $other_host 重试次数: $attempt")
        }
        Thread.sleep(1000)
      }
    } finally {
      attempt -= 1
    }
  }
  if (attempt >= 10) {
    println(s"初始化成功 ${host} ping: $ping")
  } else {
    println(s"初始化成功 ${other_host} ping: $ping")
  }
  redisClient

image.gif

修改后的程序相比最一开始鲁棒性更好,服务也更加高可用,非常的奈斯 (^-^)V

相关实践学习
基于Redis实现在线游戏积分排行榜
本场景将介绍如何基于Redis数据库实现在线游戏中的游戏玩家积分排行榜功能。
云数据库 Redis 版使用教程
云数据库Redis版是兼容Redis协议标准的、提供持久化的内存数据库服务,基于高可靠双机热备架构及可无缝扩展的集群架构,满足高读写性能场景及容量需弹性变配的业务需求。 产品详情:https://www.aliyun.com/product/kvstore     ------------------------------------------------------------------------- 阿里云数据库体验:数据库上云实战 开发者云会免费提供一台带自建MySQL的源数据库 ECS 实例和一台目标数据库 RDS实例。跟着指引,您可以一步步实现将ECS自建数据库迁移到目标数据库RDS。 点击下方链接,领取免费ECS&RDS资源,30分钟完成数据库上云实战!https://developer.aliyun.com/adc/scenario/51eefbd1894e42f6bb9acacadd3f9121?spm=a2c6h.13788135.J_3257954370.9.4ba85f24utseFl
目录
相关文章
|
25天前
|
缓存 运维 NoSQL
【Redis故障排查】「连接失败问题排查和解决」带你总体分析和整理Redis的问题故障实战开发指南及方案
【Redis故障排查】「连接失败问题排查和解决」带你总体分析和整理Redis的问题故障实战开发指南及方案
106 0
|
5天前
|
前端开发 NoSQL JavaScript
java域控连接AD遇到的问题
java域控连接AD遇到的问题
|
5天前
|
人工智能 前端开发 Java
Java语言开发的AI智慧导诊系统源码springboot+redis 3D互联网智导诊系统源码
智慧导诊解决盲目就诊问题,减轻分诊工作压力。降低挂错号比例,优化就诊流程,有效提高线上线下医疗机构接诊效率。可通过人体画像选择症状部位,了解对应病症信息和推荐就医科室。
37 10
|
7天前
|
缓存 NoSQL Java
使用Redis进行Java缓存策略设计
【4月更文挑战第16天】在高并发Java应用中,Redis作为缓存中间件提升性能。本文探讨如何使用Redis设计缓存策略。Redis是开源内存数据结构存储系统,支持多种数据结构。Java中常用Redis客户端有Jedis和Lettuce。缓存设计遵循一致性、失效、雪崩、穿透和预热原则。常见缓存模式包括Cache-Aside、Read-Through、Write-Through和Write-Behind。示例展示了使用Jedis实现Cache-Aside模式。优化策略包括分布式锁、缓存预热、随机过期时间、限流和降级,以应对缓存挑战。
|
12天前
|
运维 NoSQL 算法
Java开发-深入理解Redis Cluster的工作原理
综上所述,Redis Cluster通过数据分片、节点发现、主从复制、数据迁移、故障检测和客户端路由等机制,实现了一个分布式的、高可用的Redis解决方案。它允许数据分布在多个节点上,提供了自动故障转移和读写分离的功能,适用于需要大规模、高性能、高可用性的应用场景。
16 0
|
16天前
|
存储 缓存 NoSQL
Java手撸一个缓存类似Redis
`LocalExpiringCache`是Java实现的一个本地缓存类,使用ConcurrentHashMap存储键值对,并通过ScheduledExecutorService定时清理过期的缓存项。类中包含`put`、`get`、`remove`等方法操作缓存,并有`clearCache`方法来清除过期的缓存条目。初始化时,会注册一个定时任务,每500毫秒检查并清理一次过期缓存。单例模式确保了类的唯一实例。
13 0
|
20天前
|
NoSQL 安全 网络安全
Redis连接:加速数据访问与保障安全传输的关键
Redis连接:加速数据访问与保障安全传输的关键
|
27天前
|
缓存 NoSQL Java
Java项目:支持并发的秒杀项目(基于Redis)
Java项目:支持并发的秒杀项目(基于Redis)
26 0
|
1月前
|
Cloud Native NoSQL 数据管理
Serverless 应用引擎常见问题之首次启动获取不到redis连接如何解决
Serverless 应用引擎(Serverless Application Engine, SAE)是一种完全托管的应用平台,它允许开发者无需管理服务器即可构建和部署应用。以下是Serverless 应用引擎使用过程中的一些常见问题及其答案的汇总:
28 3
Serverless 应用引擎常见问题之首次启动获取不到redis连接如何解决
|
1月前
|
Java Maven Android开发
java如何连接mqtt
java如何连接mqtt
43 0