Redis pipeline 简介

前言

最近在学习Redis的时候看到Redis在执行命令的时候是单线程的。这意味如果客户端向服务器发送查询请求时,Redis客户端将以阻塞的方式等待服务器的响应。这样后面的客户端请求命令将不得不等待。

因此,当Redis客户端需要连续执行许多请求时,将有可能影响Redis的性能。

举个例子

客户:INCR X
服务器:1
客户:INCR X
服务器:2
客户:INCR X
服务器:3
客户:INCR X
服务器:4

Pipeline 简介

Pipe 是一种经典的设计思路。 他的主要思想为批量发送请求,批量处理请求。这样的话可以最大限度的减少网络的IO开销.
实际上,pipeline不仅可以减少网络开销,也可以减少系统调用的次数。我们知道,redis客户端写操作和读操作都会涉及 read()和write()系统调用,这意味着从用户域到内核域, 而上下文切换是巨大的速度损失。

image.png

Pipeline 代码举例

ShardedJedis shardedJedis = redisClient.borrowJedis();
        try {
            ShardedJedisPipeline pipelined = shardedJedis.pipelined();
            String key = getUserKey(userId);
            pipelined.hset(key, String.valueOf(userMissionId), Strings.EMPTY);
            pipelined.pexpire(key, EXPIRE_TIME);
            pipelined.sync();
        } finally {
            redisClient.returnJedis(shardedJedis);
 }
ShardedJedisPipeline pipelined = shardedJedis.pipelined();
 String key = getUserMissionAnswerCountKey(userMissionId);
pipelined.incr(key);
pipelined.pexpire(key, EXPIRE_TIME);
pipelined.sync();

pipeline 源码阅读

注: 下面仅代表个人阅读Redis pipeline源码的理解,不保证正确哦

pipelined()

看上去就是new了一个ShardedJedisPipeline对象,并没有做其他的事情

public ShardedJedisPipeline pipelined() {
    ShardedJedisPipeline pipeline = new ShardedJedisPipeline();
    pipeline.setShardedJedis(this);
    return pipeline;
}

pipelined.doxxx()

  • 主要是将cmd指令和指令参数写到outputStream输出流
  • 不同doXXX()都共享一个outputStream流。因此,Redis 客户端本地将维护一个outputStream流。用于写入所有的操作指令
  • 在connect()方法中对outputStream 和inputStream完成了初始化
outputStream = new RedisOutputStream(socket.getOutputStream());
inputStream = new RedisInputStream(socket.getInputStream());
protected Connection sendCommand(final Command cmd, final byte[]... args) {
    try {
      connect(); // 检查是否连接。如果已经连接则 do nothing
      Protocol.sendCommand(outputStream, cmd, args);  //将cmd指令和指令参数写到outputStream输出流
      pipelinedCommands++; //计数器+ 1 
      return this; 
    } catch (JedisConnectionException ex) {
          ...
}

sync()

  /**
   * Syncronize pipeline by reading all responses. This operation closes the pipeline. In order to
   * get return values from pipelined commands, capture the different Response<?> of the
   * commands you execute.
   */
  public void sync() {
    for (Client client : clients) {
      generateResponse(client.getOne());
    }
  }

其中generateResponse方法比较简单。

generateResponse

主要用来维护一个Response对象。并且填充Response对象 data字段的值。

//data:  可从下面的源码中看到,data 为Redis客户端与服务端之间连接的输入流
  protected Response<?> generateResponse(Object data) {
    Response<?> response = pipelinedResponses.poll();
    if (response != null) {
      response.set(data);
    }
    return response;
  }

client.getOne()

这个方法不知道为什么叫这个名字。 感觉getOne() 可能想表达的是一次性获取所有的执行结果?

  public Object getOne() {
    flush(); /// 一次性 将之前攒压的批量命令全部写入Redis server (以outputstream流的形式)
    pipelinedCommands--; //pipelinedCommands计数器 -1 
    return readProtocolWithCheckingBroken(); // 读取intputStream, 返回Redis服务器返回的结果,并赋值给Response的data属性。
  }

syncAndReturnAll

  /**
   * Syncronize pipeline by reading all responses. This operation closes the pipeline. Whenever
   * possible try to avoid using this version and use ShardedJedisPipeline.sync() as it won't go
   * through all the responses and generate the right response type (usually it is a waste of time).
   * @return A list of all the responses in the order you executed them.
   */
  public List<Object> syncAndReturnAll() {
    List<Object> formatted = new ArrayList<Object>();
    for (Client client : clients) {
      formatted.add(generateResponse(client.getOne()).get());
    }
    return formatted;
  }

syncAndReturnAll 方法和 sync方法作用类似,区别是syncAndReturnAll使用了client.getOne().get()

这个命令使得syncAndReturnAll命令自动返回了序列化的结果
get()方法主要执行BuildFactory的build方法。该方法会根据传入的泛型执行对应的序列化。 具体执行的泛型T根据接受参数类型决定:

  public static final Builder<String> STRING = new Builder<String>() {
    public String build(Object data) {
     //注意这里的encode是Redis自己写的编码方法,其实底层实现是对二进制进行decode
      return data == null ? null : SafeEncoder.encode((byte[]) data);
    }

    public String toString() {
      return "string";
    }

总结1

  • pipeline doXXX 会将本地命令批量写入outputStream
  • sync() 方法 会批量将redis命令发送到服务器,执行命令,并以inputStream二进制流的形式返回给Response 的data字段。
    • 如果需要获取Response的值,可以使用Response.get()获取
  • syncAndReturnAll 方法同样会将redis命令批量发送到服务器。同时会将redis服务器返回的结果序列化。注意,如果不需要返回结果的化,最好不要执行这个命令。而用上面的那个命令

总结2

原生批量命令与Pipeline 对比
可以使用Pipeline模拟出批量操作的效果,但是在使用时要注意它与原生批量命令的区别,主要包括如下几点:

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

推荐阅读更多精彩内容