国产精品电影_久久视频免费_欧美日韩国产激情_成年人视频免费在线播放_日本久久亚洲电影_久久都是精品_66av99_九色精品美女在线_蜜臀a∨国产成人精品_冲田杏梨av在线_欧美精品在线一区二区三区_麻豆mv在线看

基于 Netty 的 Lettuce 居然是這樣解析RESP協議的

開發 前端
既然 Lettuce 基于 Netty 實現,那么它必然在 ChannelHandler 上動手腳,直接搜索可以發現有 9 個實現類。

今天來分享 Lettuce —— 基于 Netty 實現,Springboot2 中默認的 redis 客戶端。

那它是不是直接用 Netty 中的那幾個 handler 來處理 RESP 協議的呢?一起看看吧。

可以看到這里并沒有 codec-redis 模塊,所以 Lettuce 并沒有使用 Netty 提供的 redis 模塊。

圖片圖片

(⊙﹏⊙),問題解決得太快了,那就再來思考下,它是怎么做的呢?

既然 Lettuce 基于 Netty 實現,那么它必然在 ChannelHandler 上動手腳,直接搜索可以發現有 9 個實現類。

圖片圖片

這里我關心的就是它怎么編解碼,所以直接來看 CommandEncoder 和 CommandHandler 。

打上斷點,使用測試例子直接 debug。

代碼

@Test
    void redisTest() {
        // 創建 redis 客戶端
        RedisClient redisClient = RedisClient.create("redis://123456@192.168.200.128:6379/0");
        // 創建 channel
        StatefulRedisConnection<String, String> connection = redisClient.connect();
        // 使用 sync 同步命令
        RedisCommands<String, String> syncCommands = connection.sync();

        String name = syncCommands.get("name");
        System.out.println(name);
//        syncCommands.set("key", "Hello, Redis!");

        connection.close();
        redisClient.shutdown();
    }

剛開始時,要和服務器建立連接,發送數據,涉及到 encode 流程。

CommandHandler

圖片圖片

如圖,直接來到 nioEventLoop 線程,并調用了 write 方法。

write:382, CommandHandler (io.lettuce.core.protocol)

從右邊可以看到,發了一個 HELLO 的命令出去,其中 CommandArgs 如下:

CommandArgs [buffer=$1
3
$4
AUTH
$7
default
$6
123456
]

CommandArgs?

直接來到 toString 方法,可以發現 encode 方法。

圖片圖片

如圖,有 4 個 SingularArgument:

圖片圖片

看看他們是怎么 encode 的 。

ProtocolKeywordArgument

圖片圖片

StringArgument

圖片圖片

對比 Netty

圖片圖片

貌似沒啥大的區別,可以看到 Lettuce 中,對 ByteBuf 的使用比較粗一些,Netty 中會計算這個 ByteBuf 的初始容量,而 Lettuce 就簡單些處理,直接 singularArguments.size() * 10 。

還有一個 大小端序 的處理,只能說 Netty 太細了。

圖片圖片

CommandEncoder

直接 F9 來到這一個斷點。

圖片圖片

繼續 debug ,會來到 Command 類,在這里完成對發送數據的 encode。

圖片圖片

解析下要發送的數據。

圖片圖片

小結

那么到了這里,我們就了解完 encode 的實現了。

核心:CommandArgs 中的各種 SingularArgument

圖片圖片

下面就是接受服務器數據,進行 decode 的流程了。

CommandHandler

來到 channelRead 。

圖片圖片

decode 時,會調用到 RedisStateMachine 的 decode ,它是這個流程的核心。

圖片圖片

RedisStateMachine?

Redis 狀態機:

圖片圖片

這里我直接 copy 了一份 。

static class State {

    // Callback interface to handle a {@link State}.
    @FunctionalInterface
    interface StateHandler {
        Result handle(RedisStateMachine rsm, State state, ByteBuf buffer, CommandOutput<?, ?, ?> output,
                Consumer<Exception> errorHandler);
    }

    enum Type implements StateHandler {

        SINGLE('+', RedisStateMachine::handleSingle),

        ERROR('-', RedisStateMachine::handleError),

        INTEGER(':', RedisStateMachine::handleInteger),

        // 下面開始都是 @since 6.0/RESP3
        FLOAT(',', RedisStateMachine::handleFloat),

        BOOLEAN('#', RedisStateMachine::handleBoolean),

        BULK_ERROR('!', RedisStateMachine::handleBulkError),

        VERBATIM('=', RedisStateMachine::handleBulkAndVerbatim), VERBATIM_STRING('=', RedisStateMachine::handleVerbatim),

        BIG_NUMBER('(', RedisStateMachine::handleBigNumber),

        MAP('%', RedisStateMachine::handleMap),

        SET('~', RedisStateMachine::handleSet),

        ATTRIBUTE('|', RedisStateMachine::handleAttribute),

        PUSH('>', RedisStateMachine::handlePushAndMulti),
       
        HELLO_V3('@', RedisStateMachine::handleHelloV3),

        NULL('_', RedisStateMachine::handleNull),

        BULK('$', RedisStateMachine::handleBulkAndVerbatim),

        MULTI('*', RedisStateMachine::handlePushAndMulti), BYTES('*', RedisStateMachine::handleBytes);

        final byte marker;

        private final StateHandler behavior;

        Type(char marker, StateHandler behavior) {
            this.marker = (byte) marker;
            this.behavior = behavior;
        }

        @Override
        public Result handle(RedisStateMachine rsm, State state, ByteBuf buffer, CommandOutput<?, ?, ?> output,
                Consumer<Exception> errorHandler) {
            return behavior.handle(rsm, state, buffer, output, errorHandler);
        }
    }

    enum Result {
        NORMAL_END, BREAK_LOOP, CONTINUE_LOOP
    }

    Type type = null;

    int count = NOT_FOUND;

    @Override
    public String toString() {
        final StringBuffer sb = new StringBuffer();
        sb.append(getClass().getSimpleName());
        sb.append(" [type=").append(type);
        sb.append(", count=").append(count);
        sb.append(']');
        return sb.toString();
    }

}

繼續 debug,會來到 doDecode 方法。

這里有兩個核心步驟:

  1. 根據讀取到的第一個字節,判斷是不是 RESP3。
  2. 調用 狀態機 中的 State.Type 枚舉類,處理 handle。

圖片

這里先手動解析下服務器返回的數據。

ByteBufUtil.decodeString(buffer,0,146, Charset.defaultCharset());
%7
$6
server
$5
redis
$7
version
$6
6.0.12
$5
proto
:3
$2
id
:74
$4
mode
$10
standalone
$4
role
$6
master
$7
modules
*0

handleMap

%7 對應的 handler 處理。

圖片圖片

后面就進入 狀態機 流程判斷了,上面我們拿到的數據要循環好久,就不一一列舉出來了。

$6 對應的 handler 處理。

圖片圖片

最后解析出來剛好 7 個,可以對比上面手動解析的結果驗證下。

圖片圖片

小結

到了這里,decode 的流程也完畢了,畫個圖總結下??。

圖片圖片

結尾

Lettuce 的 decode 依賴于 狀態機 RedisStateMachine 實現,encode 靠 SingularArgument 實現。

圖片圖片

這次我做了兩種嘗試:

  1. 按以往的方式,從測試例子開始 debug。
  2. 思考下框架的特性,直奔主題。

兩種方式都收獲頗豐,但第二種嘗試得比較少,以后可以多多實踐,站在不同的角度去思考問題。

責任編輯:武曉燕 來源: Java4ye
相關推薦

2024-05-16 07:55:54

NettyRedisRESP協議

2019-08-09 10:15:07

程序員項目研發

2021-09-29 00:19:10

容器集群k8s

2021-01-22 09:11:34

Python多線程CPU

2025-07-21 08:22:30

localhost身份證號綽號

2021-08-02 15:06:46

vim服務Java

2022-01-12 19:59:19

Netty 核心啟動

2025-03-12 10:36:32

2021-06-02 16:19:14

技術研發指標

2020-10-26 16:35:53

內存JavaThreadLocal

2014-11-11 09:56:54

2013-09-02 09:44:54

2021-10-08 09:07:09

算法程序技術

2018-01-30 11:52:39

IDC全閃存

2020-10-09 14:46:57

阿里巴巴互聯網存儲

2017-06-12 17:47:19

2020-08-10 10:59:00

黑客?推特漏洞

2018-08-03 09:26:06

2015-10-13 10:49:44

Pear OSMac OS XLinux

2021-01-04 15:02:21

加密貨幣區塊鏈存儲
點贊
收藏

51CTO技術棧公眾號

国产日韩一区二区三区在线播放 | 5g影院天天爽成人免费下载| 在线免费观看av的网站| 亚洲精品无吗| 欧美xxxxxxxx| 濑亚美莉一二区在线视频 | 久久亚洲精品毛片| 77导航福利在线| 欧美高清在线一区| 天堂va久久久噜噜噜久久va| 久久久久av| 羞羞网www| 狠狠色丁香婷综合久久| 成人黄色av网| 久久亚州av| 中文字幕视频一区二区在线有码| 麻豆传媒视频在线观看| 精品久久久久久久中文字幕| 拍拍拍999自拍偷| av在线一区二区三区| 日本福利视频导航| 久久高清免费观看| av一区和二区| 一区二区三区毛片免费| 国产精品揄拍500视频| 卡通动漫国产精品| 欧美成aaa人片免费看| 成人av免费电影网站| 欧美成人激情免费网| 成人午夜电影在线观看| 日韩欧美一区视频| 欧美vide| 精品国产31久久久久久| 欧美vide| 狠狠色狠狠色综合日日五| 最新天堂资源在线资源| 午夜精品成人在线视频| 中文字幕在线观看第一页| 亚洲一区在线视频观看| wwwcom羞羞网站| 欧美国产日韩精品免费观看| 9久久婷婷国产综合精品性色| 99在线热播精品免费| 少妇无码av无码专区在线观看 | 蜜桃麻豆www久久国产精品| 亚洲男女av一区二区| 成人性生交大片免费看小说| 亚洲最新av| 蜜桃成人在线| 免费一区二区视频| www.18av.com| 国产亚洲制服色| 日韩一区二区三区不卡视频| 成人免费小视频| 成人亚洲一区二区三区| 亚洲一区二区av电影| 在线看视频你懂得| 欧美三级日韩三级| 日本高清成人vr专区| 亚洲一区999| 给我免费播放日韩视频| 国产欧美日韩免费| 亚洲国内欧美| 在线一区亚洲| 亚洲国产激情av| 羞羞视频在线观看| 欧美sm极限捆绑bd| 国产午夜亚洲精品一级在线| 欧美中文字幕在线观看| 欧美性久久久| 女同性恋一区二区| 国产欧美日韩中文久久| 国产香蕉视频在线观看| 56国语精品自产拍在线观看| 婷婷午夜社区一区| 欧美中文字幕精品| 国产婷婷精品| 国产免费毛卡片| 亚洲丰满少妇videoshd| 免费毛片在线看片免费丝瓜视频 | japanese在线播放| 国产精品入口麻豆九色| 日夜干在线视频| 亚洲成人精品av| 亚洲91在线| 97免费资源站| 成人午夜av电影| 在线免费观看黄色片| 亚洲精品一区二区三区不| 西瓜成人精品人成网站| 人偷久久久久久久偷女厕| 国产午夜精品一区二区| 香蕉视频免费在线播放| 欧美乱大交xxxxx另类电影| 欧美高清不卡| 成人免费观看毛片| 7777精品伊人久久久大香线蕉完整版 | 日本在线播放一区| 不卡一区二区三区四区| 最新91在线| 亚洲欧美日韩中文在线制服| 色97色成人| 黄色一级视频在线播放| 色狠狠av一区二区三区| 一级欧美视频| 日韩av电影在线观看| 亚洲日本va午夜在线影院| 欧美专区日韩专区| 大胆人体一区二区| 91情侣偷在线精品国产| 99re视频精品| 久久五月精品| 日韩av免费在线播放| 国产成人精品综合在线观看| 欧美日韩在线中文字幕| 欧美激情精品久久久久久大尺度| 香蕉久久久久久久av网站| gay视频丨vk| 综合网日日天干夜夜久久| 99热这里只有成人精品国产| av小说在线| 久久艳片www.17c.com| 青椒成人免费视频| 国产黄色在线播放| 亚洲激情图片一区| 精品一区二区三区免费观看| 免费欧美电影| 羞羞在线视频| 97久久国产亚洲精品超碰热| 懂色一区二区三区免费观看| 日本一区高清| 欧美激情一区二区三区在线视频观看| 日韩中文字幕亚洲一区二区va在线| 一二三区高清| 久久久久久久久91| 国产成人综合在线| 色呦呦在线视频| 国产乱码精品一区二区三区卡| 一区二区视频在线| 久久男人av| www.99com| 午夜精品在线视频| 2020国产精品自拍| 欧美天堂一区二区| 久久久久久av无码免费网站下载| 欧美一区二区视频在线观看2022| 91精品国产视频| 少妇**av毛片在线看| 4444欧美成人kkkk| 亚洲欧洲精品天堂一级| www.国产精品一区| 成人免费淫片95视频观看网站| 欧美国产日韩一区二区| 99精品国产热久久91蜜凸| 欧美成人性网| 人人妻人人澡人人爽欧美一区双 | 丝袜美腿高跟呻吟高潮一区| 久蕉依人在线视频| 亚洲淫片在线视频| 欧美性xxxx极品hd满灌| 欧美黄色录像片| 亚洲成人动漫在线播放| 亚洲精品999| 在线视频你懂得一区二区三区| 色视频免费在线观看| 国产91亚洲精品| 亚洲国产精品自拍| 欧美精品一区二区三区中文字幕| 国产精品一区二区小说| 91黑丝高跟在线| 亚洲美女屁股眼交| 精品久久中文| 久久精品国产亚洲a∨麻豆| 一本一道久久a久久精品| 亚洲精品视频网上网址在线观看| 日韩中文有码在线视频| 国产一区日韩| 最近中文视频在线| 国产精自产拍久久久久久蜜| 欧美日韩一区二区免费在线观看| 欧美 日韩 国产一区二区在线视频 | 日韩不卡免费视频| 丁香花视频在线观看| 国产一二三四五| 久久久精品国产一区二区| 久久精品免视看| 欧美精品一区二区三区中文字幕 | 亚洲国产精品成人一区二区| 国产 欧美在线| 99a精品视频在线观看| 黄色高清在线观看| 欧美高清性xxxxhd| 免费亚洲一区| 色资源二区在线视频| 免费无码国产v片在线观看| 热99精品只有里视频精品| 在线亚洲精品福利网址导航| 国产中文字幕一区| 国产精品99久久久久久人 | 久久成人福利|