Java自学者论坛

 找回密码
 立即注册

手机号码,快捷登录

恭喜Java自学者论坛(https://www.javazxz.com)已经为数万Java学习者服务超过8年了!积累会员资料超过10000G+
成为本站VIP会员,下载本站10000G+会员资源,会员资料板块,购买链接:点击进入购买VIP会员

JAVA高级面试进阶训练营视频教程

Java架构师系统进阶VIP课程

分布式高可用全栈开发微服务教程Go语言视频零基础入门到精通Java架构师3期(课件+源码)
Java开发全终端实战租房项目视频教程SpringBoot2.X入门到高级使用教程大数据培训第六期全套视频教程深度学习(CNN RNN GAN)算法原理Java亿级流量电商系统视频教程
互联网架构师视频教程年薪50万Spark2.0从入门到精通年薪50万!人工智能学习路线教程年薪50万大数据入门到精通学习路线年薪50万机器学习入门到精通教程
仿小米商城类app和小程序视频教程深度学习数据分析基础到实战最新黑马javaEE2.1就业课程从 0到JVM实战高手教程MySQL入门到精通教程
查看: 426|回复: 0

一次线上Redis类转换异常排查引发的思考

[复制链接]
  • TA的每日心情
    奋斗
    2024-11-24 15:47
  • 签到天数: 804 天

    [LV.10]以坛为家III

    2053

    主题

    2111

    帖子

    72万

    积分

    管理员

    Rank: 9Rank: 9Rank: 9

    积分
    726782
    发表于 2021-5-28 14:48:37 | 显示全部楼层 |阅读模式

    之前同事反馈说线上遇到Redis反序列化异常问题,异常如下:

    XxxClass1 cannot be cast to XxxClass2

    已知信息如下:

    • 该异常不是必现的,偶尔才会出现;
    • 出现该异常后重启应用或者过一会就好了;
    • 序列化协议使用了hessian。

    因为偶尔出现,首先看了报异常那块业务逻辑是不是有问题,看了一遍也发现什么问题。看了下对应日志,发现是在Redis读超时之后才出现的该异常,因此怀疑redis client操作逻辑那块导致的(公司架构组对redis做了一层封装),发现获取/释放redis连接如下代码:

     1 try {
     2     jedis = jedisPool.getResource();
     3     // jedis业务读写操作
     4 } catch (Exception e) {
     5     // 异常处理
     6 } finally {
     7     if (jedis != null) {
     8         // 归还给连接池
     9         jedisPool.returnResourceObject(jedis);
    10     }
    11 }

    初步认定原因为:发生了读写超时的连接,直接归还给连接池,下次使用该连接时读取到了上一次Redis返回的数据。因此本地验证下,示例代码如下:

     1 @Data
     2 @NoArgsConstructor
     3 @AllArgsConstructor
     4 static class Person implements Serializable {
     5     private String name;
     6     private int age;
     7 }
     8 @Data
     9 @NoArgsConstructor
    10 @AllArgsConstructor
    11 static class Dog implements Serializable {
    12     private String name;
    13 }
    14 
    15 public static void main(String[] args) throws Exception {
    16     JedisPoolConfig config = new JedisPoolConfig();
    17     config.setMaxTotal(1);
    18     JedisPool jedisPool = new JedisPool(config, "192.168.193.133", 6379, 2000, "123456");
    19 
    20     Jedis jedis = jedisPool.getResource();
    21     jedis.set("key1".getBytes(), serialize(new Person("luoxn28", 26)));
    22     jedis.set("key2".getBytes(), serialize(new Dog("tom")));
    23     jedisPool.returnResourceObject(jedis);
    24 
    25     try {
    26         jedis = jedisPool.getResource();
    27         Person person = deserialize(jedis.get("key1".getBytes()), Person.class);
    28         System.out.println(person);
    29     } catch (Exception e) {
    30         // 发生了异常之后,未对该连接做任何处理
    31         System.out.println(e.getMessage());
    32     } finally {
    33         if (jedis != null) {
    34             jedisPool.returnResourceObject(jedis);
    35         }
    36     }
    37 
    38     try {
    39         jedis = jedisPool.getResource();
    40         Dog dog = deserialize(jedis.get("key2".getBytes()), Dog.class);
    41         System.out.println(dog);
    42     } catch (Exception e) {
    43         System.out.println(e.getMessage());
    44     } finally {
    45         if (jedis != null) {
    46             jedisPool.returnResourceObject(jedis);
    47         }
    48     }
    49 }

    连接超时时间设置2000ms,为了方便测试,可以在redis服务器上使用gdb命令断住redis进程(如果redis部署在Linux系统上的话,还可以使用iptable命令在防火墙禁止某个回包),比如在执行 jedis.get("key1".getBytes() 代码前,对redis进程使用gdb命令断住,那么就会导致读取超时,然后就会触发如下异常:

    Person cannot be cast to Dog

    既然已经知道了该问题原因并且本地复现了该问题,对应解决方案是,在发生异常时归还给连接池时关闭该连接即可(jedis.close内部已经做了判断),代码如下:

     1 try {
     2     jedis = jedisPool.getResource();
     3     // jedis业务读写操作
     4 } catch (Exception e) {
     5     // 异常处理
     6 } finally {
     7     if (jedis != null) {
     8         // 归还给连接池
     9         jedis.close();
    10     }
    11 }

    至此,该问题解决。注意,因为使用了hessian序列化(其包含了类型信息,类似的有Java本身序列化机制),所有会报类转换异常;如果使用了json序列化(其只包含对象属性信息),反序列化时不会报异常,只不过因为不同类的属性不同,会导致反序列化后的对象属性为空或者属性值混乱,使用时会导致问题,并且这种问题因为没有报异常所以更不容易发现。

     

    既然说到了Redis的连接,要知道的是,Redis基于RESP(Redis Serialization Protocol)协议来通信,并且通信方式是停等方式,也就说一次通信独占一个连接直到client读取到返回结果之后才能释放该连接让其他线程使用。小伙伴们可以思考一下,Redis通信能否像dubbo那样使用单连接+序列号(标识单次通信)通信方式呢?理论上是可以的,不过由于RESP协议中并没有一个"序列号"的字段,所以直接靠原生的通信方法来实现是不现实的。不过我们可以通过echo命令传递并返回"序列号"+正常的读写方式来实现,这里要保证二者执行的原子性,可以通过lua脚本或者事务来实现,事务方式如下:

    MULTI
    ECHO "唯一序列号"
    GET key1
    EXEC

    然后客户端收到的结果是一个 [ "唯一序列号", "value1" ]的列表,你可以根据前一项识别出这是你发送的哪个请求。

    为什么Redis通信方式并没有采用类似于dubbo这种通信方式呢,个人认为有以下几点:

    • 使用停等这种通信方式实现简单,并且协议字段尽可能紧凑;
    • Redis都是内存操作,处理性能较强,停等协议不会造成客户端等待时间较长;
    • 目前来看,通信方式这块不是Redis使用上的性能瓶颈,这一点很重要。

     

    推荐阅读:

     欢迎小伙伴扫描以下二维码阅读更多精彩好文。

     

    哎...今天够累的,签到来了1...
    回复

    使用道具 举报

    您需要登录后才可以回帖 登录 | 立即注册

    本版积分规则

    QQ|手机版|小黑屋|Java自学者论坛 ( 声明:本站文章及资料整理自互联网,用于Java自学者交流学习使用,对资料版权不负任何法律责任,若有侵权请及时联系客服屏蔽删除 )

    GMT+8, 2025-1-23 12:18 , Processed in 0.055626 second(s), 27 queries .

    Powered by Discuz! X3.4

    Copyright © 2001-2021, Tencent Cloud.

    快速回复 返回顶部 返回列表