成人免费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-01-22 09:11:34

Python多線程CPU

2021-09-29 00:19:10

容器集群k8s

2025-03-12 10:36:32

2021-06-02 16:19:14

技術研發指標

2021-08-02 15:06:46

vim服務Java

2022-01-12 19:59:19

Netty 核心啟動

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

2021-01-04 15:02:21

加密貨幣區塊鏈存儲

2020-08-10 10:59:00

黑客?推特漏洞

2021-04-29 14:57:16

惡意軟件黑客網絡攻擊

2015-10-13 10:49:44

Pear OSMac OS XLinux
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 成人午夜电影网 | 国产精品视频久久 | 中文字幕一区二区三区精彩视频 | 天天干天天爽 | 中文字幕精品一区 | 天天综合日日夜夜 | 精品国产一区二区在线 | 伊人网99| 国产三级精品三级在线观看四季网 | 国产美女久久久 | 成人免费精品视频 | 精品久久国产视频 | 日韩精品在线一区 | 一级黄a| 久久久国产一区二区三区四区小说 | 中文字幕91av | 拍拍无遮挡人做人爱视频免费观看 | 亚洲成人动漫在线观看 | 国产精品欧美一区二区三区不卡 | 国产999精品久久久 精品三级在线观看 | 中文字幕国产视频 | 亚洲一区二区电影在线观看 | 成人在线免费观看视频 | 日韩有码一区二区三区 | 国产线视频精品免费观看视频 | 亚洲欧美在线观看视频 | 成人性视频免费网站 | 91n成人| 自拍偷拍亚洲一区 | www.久久久久久久久久久久 | 99久久久久久久 | 久久久久综合 | 亚洲国产精品第一区二区 | 99热精品在线观看 | 免费的黄色片子 | 最新日韩欧美 | 欧美午夜精品理论片a级按摩 | 天天色天天色 | 亚洲国产第一页 | 日韩欧美中文 | 国产免费一级片 |