redis訂閱發(fā)布

問題描述

在redis集成到spring后吹零, 使用了redis的訂閱與發(fā)布的功能晰房,在每次發(fā)布的時候, 會出現(xiàn)瘋狂日志報錯窍荧。 由于該問題比較隱蔽辉巡,在這邊記錄并提供解決辦法。

集成說明

  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
                }
            }
        }
    }

}

/**
 * 具體的執(zhí)行邏輯
 */
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不一樣输硝, 并且應(yīng)用場景也不一樣今瀑, 比如對于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的訂閱發(fā)布功能, 如何使用呢, 請看下面代碼


@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方法毛仪, 導(dǎo)致每次生產(chǎn)發(fā)布后芯勘, 在日志中都會出現(xiàn)大量druid獲取的連接已關(guān)閉的錯誤, 然而我發(fā)現(xiàn)服務(wù)器上的業(yè)務(wù)時正常運(yùn)行的衡怀。

是什么原因?qū)е逻@個問題的產(chǎn)生呢抛杨?

經(jīng)過一定的校驗我發(fā)現(xiàn), 在發(fā)布生產(chǎn)時玉罐,數(shù)據(jù)源會被關(guān)閉吊输, 然而redis客戶端還在訂閱服務(wù)端的chennel,此時服務(wù)器又啟動透硝,在這個類初始化之前, 繼而調(diào)用init函數(shù),此時又會有一個redis客戶端訂閱該channe

簡單畫一個圖解:

解決辦法

解決辦法正如我上面寫的, 寫一個destry方法浴井, 在該對象銷毀之前洪囤, 取消訂閱channel即可。 這樣始終保持每次發(fā)布訂閱一個channel锦溪, 結(jié)束銷毀一個channel牺丙。

其他注意點(diǎn):

還會發(fā)現(xiàn), 我們這邊在訂閱的時候?qū)懥藗€線程肖揣,原因是因為redis訂閱是同步的, 其實我們通過jedis源碼很容易看到, 在內(nèi)部是通過一個do while循環(huán)來監(jiān)聽消息。所以我們將該訂閱交給一個線程執(zhí)行一屋, 使得應(yīng)用能夠正常被啟動诽嘉。

如何提高消息的處理能力呢?
很簡單,我們可以在onMessage中加入線程池來處理消息。

spring-data-redis

spring-data-redis為我們在jedis做了更好的封裝, 使得我們對redis客戶端能夠很簡單的應(yīng)用宙帝, 下面簡單介紹以下spring-data-redis的發(fā)布與訂閱功能步脓。
其中我們只需要進(jìn)行相關(guān)的bean的配置要出, 無需關(guān)心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;
    }


}

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市囱挑,隨后出現(xiàn)的幾起案子弹惦,更是在濱河造成了極大的恐慌棠隐,老刑警劉巖嗡贺,帶你破解...
    沈念sama閱讀 211,290評論 6 491
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件诫睬,死亡現(xiàn)場離奇詭異亲澡,居然都是意外死亡客情,警方通過查閱死者的電腦和手機(jī)概页,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,107評論 2 385
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來悯仙,“玉大人锡垄,你說我怎么就攤上這事∩η矗” “怎么了垮卓?”我有些...
    開封第一講書人閱讀 156,872評論 0 347
  • 文/不壞的土叔 我叫張陵灭将,是天一觀的道長。 經(jīng)常有香客問我后控,道長庙曙,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 56,415評論 1 283
  • 正文 為了忘掉前任浩淘,我火速辦了婚禮捌朴,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘张抄。我一直安慰自己砂蔽,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,453評論 6 385
  • 文/花漫 我一把揭開白布署惯。 她就那樣靜靜地躺著左驾,像睡著了一般。 火紅的嫁衣襯著肌膚如雪极谊。 梳的紋絲不亂的頭發(fā)上诡右,一...
    開封第一講書人閱讀 49,784評論 1 290
  • 那天,我揣著相機(jī)與錄音轻猖,去河邊找鬼帆吻。 笑死,一個胖子當(dāng)著我的面吹牛咙边,可吹牛的內(nèi)容都是我干的桅锄。 我是一名探鬼主播琉雳,決...
    沈念sama閱讀 38,927評論 3 406
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼友瘤!你這毒婦竟也來了翠肘?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 37,691評論 0 266
  • 序言:老撾萬榮一對情侶失蹤辫秧,失蹤者是張志新(化名)和其女友劉穎束倍,沒想到半個月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體盟戏,經(jīng)...
    沈念sama閱讀 44,137評論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡绪妹,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,472評論 2 326
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了柿究。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片邮旷。...
    茶點(diǎn)故事閱讀 38,622評論 1 340
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖蝇摸,靈堂內(nèi)的尸體忽然破棺而出婶肩,到底是詐尸還是另有隱情,我是刑警寧澤貌夕,帶...
    沈念sama閱讀 34,289評論 4 329
  • 正文 年R本政府宣布律歼,位于F島的核電站,受9級特大地震影響啡专,放射性物質(zhì)發(fā)生泄漏险毁。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,887評論 3 312
  • 文/蒙蒙 一们童、第九天 我趴在偏房一處隱蔽的房頂上張望畔况。 院中可真熱鬧,春花似錦慧库、人聲如沸问窃。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,741評論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至嵌戈,卻和暖如春覆积,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背熟呛。 一陣腳步聲響...
    開封第一講書人閱讀 31,977評論 1 265
  • 我被黑心中介騙來泰國打工宽档, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人庵朝。 一個月前我還...
    沈念sama閱讀 46,316評論 2 360
  • 正文 我出身青樓吗冤,卻偏偏與公主長得像又厉,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子椎瘟,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,490評論 2 348

推薦閱讀更多精彩內(nèi)容

  • MVC Model:模型 View:視圖 Controller:控制器 單例 單例使用懶加載方式在第一次實例時創(chuàng)建...
    lucifrom_long閱讀 399評論 0 1
  • 1 那年覆致,她還是個家庭主婦。 懷里抱著一個小小的孩子肺蔚,粉妝玉琢煌妈。她對這個孩子怎么也愛不夠,整天抱在懷里宣羊,看著孩子的...
    亦暖橙閱讀 406評論 1 0
  • 介紹 Hotline Miami 是什么璧诵,以及它為什么這么棒。 引言 我總覺得獨(dú)立游戲的發(fā)展?fàn)顩r跟我所認(rèn)知的獨(dú)立音...
    jagttt閱讀 6,124評論 2 7