redis订阅发布

问题描述

在redis集成到spring后, 使用了redis的订阅与发布的功能,在每次发布的时候, 会出现疯狂日志报错。 由于该问题比较隐蔽,在这边记录并提供解决办法。

集成说明

  1. spring 版本: 4.1.4
  2. redis客户端: jedis 2.7.2
  3. 连接池druid
    使用application.xml配置redis连接池等的信息。

对jedis的简单封装

在项目中, 我对jedis客户端做了简单的封装, 封装代码如下:


@Repository("redisClientTemplate")
public class RedisClientTemplate {

    @Autowired
    private ShardedJedisPool shardedJedisPool;
    @Autowired
    private JedisPool jedisPool;
    /**
     *
     * @param ec
     * @return
     * @throws IllegalArgumentException when ec is null
     * @throws RuntimeException
     */

    public <T>T execute(Executor<T> ec, Class<? extends JedisCommands> jedisClass){
        Assert.notNull(ec, "the redis executor is null");
        JedisCommands jedisCommands = null;
        try {
            if (jedisClass == Jedis.class) {
                jedisCommands = jedisPool.getResource();
            } else {
                jedisCommands = shardedJedisPool.getResource();
            }
            return ec.execute(jedisCommands);
        }catch (Exception e){
            throw new RuntimeException(e);
        }finally {
            if (jedisCommands!=null && jedisCommands instanceof Closeable){
                try {
                    ((Closeable)jedisCommands).close();
                } catch (IOException e) {
                    //  log
                }
            }
        }
    }

}

/**
 * 具体的执行逻辑
 */
public abstract class Executor<T> {

    T execute(JedisCommands jedisCommands){
        if (jedisCommands instanceof Jedis){
            return doExecute((Jedis)jedisCommands);
        }else if (jedisCommands instanceof ShardedJedis){
            return doExecute((ShardedJedis)jedisCommands);
        }else {
            return doExecute((ShardedJedis)jedisCommands);
        }
    }

    protected T doExecute(ShardedJedis jedisCommands) {
        return null;
    }

    protected T doExecute(Jedis jedisCommands) {
        return null;
    }

}


在这里说明一下, 由于jedis存在jedis和shardedJedis, 2种api不一样, 并且应用场景也不一样, 比如对于key的模糊搜索ShardedJedis此操作。

使用example:

redisClientTemplate.execute(ExecutorUtils.addSet(key, value), ShardedJedis.class);

在ExecutorUtils
public static Executor<Boolean> addSet(final String key, final String... value){
        if (key == null || key.equals("")) {
            return null;  // todo illegalArgument
        }
        return new Executor<Boolean>() {
            @Override
            public Boolean doExecute(ShardedJedis shardedJedis) {
                Long i = shardedJedis.sadd(key, value); // todo test
                return true;
            }
        };
    }

jedis的api 使用起来很简单, 在项目中, 使用了redis的订阅发布功能, 如何使用呢, 请看下面代码


@Component
@Slf4j
public class SubscriberDemo extends JedisPubSub {
    Gson gson = new Gson();

    @Autowired
    RedisClientTemplate redisClientTemplate;

    @Override
    public void onMessage(String channel, String message) {
        doSomethingWithDatabase();
    }
    
    @PostConstruct
    public void init (){
        new Thread(new Runnable() {
            @Override
            public void run() {
                redisClientTemplate.execute(new Executor<Object>() {
                    @Override
                    protected Object doExecute(Jedis jedisCommands) {
                        jedisCommands.subscribe(SubscriberDemo.this,"channel");
                        return null;
                    }
                }, Jedis.class);
            }
        },"threadName").start();

    }
    @PreDestroy
    public void destroy(){
        this.unsubscribe("channel");
    }
}

问题详情

在使用jedis时,我并没有加入destroy方法, 导致每次生产发布后, 在日志中都会出现大量druid获取的连接已关闭的错误, 然而我发现服务器上的业务时正常运行的。

是什么原因导致这个问题的产生呢?

经过一定的校验我发现, 在发布生产时,数据源会被关闭, 然而redis客户端还在订阅服务端的chennel,此时服务器又启动,在这个类初始化之前, 继而调用init函数,此时又会有一个redis客户端订阅该channe

简单画一个图解:

解决办法

解决办法正如我上面写的, 写一个destry方法, 在该对象销毁之前, 取消订阅channel即可。 这样始终保持每次发布订阅一个channel, 结束销毁一个channel。

其他注意点:

还会发现, 我们这边在订阅的时候写了个线程,原因是因为redis订阅是同步的, 其实我们通过jedis源码很容易看到, 在内部是通过一个do while循环来监听消息。所以我们将该订阅交给一个线程执行, 使得应用能够正常被启动。

如何提高消息的处理能力呢?
很简单,我们可以在onMessage中加入线程池来处理消息。

spring-data-redis

spring-data-redis为我们在jedis做了更好的封装, 使得我们对redis客户端能够很简单的应用, 下面简单介绍以下spring-data-redis的发布与订阅功能。
其中我们只需要进行相关的bean的配置, 无需关心channel的订阅

@Configuration
public class RedisSubListenerConfig {
    final ExecutorService executorService = Executors.newFixedThreadPool(20);
    @Bean
    RedisMessageListenerContainer container(RedisConnectionFactory connectionFactory,
                                            MessageListenerAdapter service) {
        RedisMessageListenerContainer container = new RedisMessageListenerContainer();
        container.setConnectionFactory(connectionFactory);
        container.addMessageListener(dataWarning, new PatternTopic(TOPIC));
        return container;
    }

    @Bean
    MessageListenerAdapter dataAdapter(Service service) {
        MessageListenerAdapter adapter = new MessageListenerAdapter(service, "doSomething");
        adapter.setSerializer(new FastJsonRedisSerializer<>(Message.class));
        return adapter;
    }


}

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,172评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,346评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,788评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,299评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,409评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,467评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,476评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,262评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,699评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,994评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,167评论 1 343
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,827评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,499评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,149评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,387评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,028评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,055评论 2 352

推荐阅读更多精彩内容

  • MVC Model:模型 View:视图 Controller:控制器 单例 单例使用懒加载方式在第一次实例时创建...
    lucifrom_long阅读 402评论 0 1
  • 1 那年,她还是个家庭主妇。 怀里抱着一个小小的孩子,粉妆玉琢。她对这个孩子怎么也爱不够,整天抱在怀里,看着孩子的...
    亦暖橙阅读 408评论 1 0
  • 介绍 Hotline Miami 是什么,以及它为什么这么棒。 引言 我总觉得独立游戏的发展状况跟我所认知的独立音...
    jagttt阅读 6,139评论 2 7