成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

基于 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

2021-08-02 15:06:46

vim服務Java

2025-03-12 10:36:32

2022-01-12 19:59:19

Netty 核心啟動

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

2018-08-03 09:26:06

2020-08-10 10:59:00

黑客?推特漏洞

2015-10-13 10:49:44

Pear OSMac OS XLinux

2021-01-04 15:02:21

加密貨幣區塊鏈存儲

2021-04-29 14:57:16

惡意軟件黑客網絡攻擊
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: www.久久久久久久久久久 | 国产精品久久久久久婷婷天堂 | 亚洲国产成人精品在线 | 麻豆av在线免费观看 | 成人一级片在线观看 | 成人精品视频99在线观看免费 | 91亚洲精选 | 精品视频在线一区 | 亚洲高清免费观看 | 天堂在线www | 99久久婷婷国产综合精品电影 | 亚洲一二三区精品 | 国产高清视频在线观看 | 久久久国产一区二区三区 | 中文字幕在线观看一区 | 91久久北条麻妃一区二区三区 | 日韩精品一区二区久久 | 午夜噜噜噜 | www.久久99 | 亚洲一区| 黄网免费看 | 国产a视频 | 久久久无码精品亚洲日韩按摩 | 资源首页二三区 | 日韩精品一区二区三区在线观看 | 色999日韩 | 国产精品高清在线 | 天天操夜夜操 | 日韩成人在线免费视频 | 国产精产国品一二三产区视频 | 日韩成人在线视频 | 免费99精品国产自在在线 | 国产一区二区三区在线视频 | 91亚洲国产亚洲国产 | 成人一区av偷拍 | 中国免费黄色片 | 91精品久久久久久久99 | 国产欧美精品一区二区三区 | 免费看的黄网站 | 久久免费国产 | 久久久精品网 |