kafka 消费者详解

前言

读完本文,你将了解到如下知识点:

  1. kafka 的消费者 和 消费者组
  2. 如何正确使用 kafka consumer
  3. 常用的 kafka consumer 配置

消费者 和 消费者组

  1. 什么是消费者?
    顾名思义,消费者就是从kafka集群消费数据的客户端,
    如下图,展示了一个消费者从一个topic中消费数据的模型
    图1
  1. 单个消费者模型存在的问题?
    如果这个时候 kafka 上游生产的数据很快,
    超过了这个消费者1 的消费速度,
    那么就会导致数据堆积,
    产生一些大家都知道的蛋疼事情了,
    那么我们只能加强 消费者 的消费能力,
    所以也就有了我们下面来说的 消费者组

  2. 什么是消费者组?
    所谓 消费者组,其实就是一组 消费者 的集合,
    当我们看到下面这张图是不是就特别舒服了,
    我们采用了一个 消费组 来消费这个 topic
    众人拾柴火焰高,其消费能力那是按倍数递增的,
    所以这里我们一般来说都是采用 消费者组 来消费数据,
    而不会是 单消费者 来消费数据的。
    这里值得我们注意的是:

    • 一个topic 可以被 多个 消费者组 消费,
      但是每个 消费者组 消费的数据是 互不干扰 的,
      也就是说,每个 消费组 消费的都是 完整的数据
    • 一个分区只能被 同一个消费组内 的一个 消费者 消费,
      不能拆给多个消费者 消费,
      也就是说如果你某个 消费者组内的消费者数 比 该 Topic 的分区数还多,
      那么多余的消费者是不起作用的
图2
  1. 是不是一个 消费组 的 消费者 越多其消费能力就越强呢?
    图3 我们就很好的可以回答这个问题了,
    我们可以看到 消费者4 是完全没有消费任何的数据的,
    所以如果你想要加强 消费者组 的能力,
    除了添加消费者,分区的数量也是需要跟着增加的
    只有这样他们的并行度才能上的去,消费能力才会强。
图3
  1. 为了提高 消费组 的 消费能力,我是不是可以随便添加 分区 和 消费者 呢?
    答案当然是否定的啦。。。嘿嘿
    我们看到图2,一般来说我们建议 消费者 数量 和 分区 数量是一致的
    当我们的消费能力不够时,
    就必须通过调整分区的数量来提高并行度,
    但是,我们应该尽量来避免这种情况发生,
    比如:
    现在我们需要在图2的基础上增加一个 分区4
    那么这个 分区4 该由谁来消费呢?
    这个时候kafka会进行 分区再均衡
    来为这个分区分配消费者,分区再均衡 期间该 Topic 是不可用的,
    并且作为一个 被消费者
    分区数的改动将影响到每一个消费者组
    所以在创建 topic 的时候,我们就应该考虑好分区数
    来尽量避免这种情况发生

    这里我们额外补充一点关于如何避免 分区再均衡 的知识,
    这里主要补充的是生产环境中因为不正确的配置引起的不需要的 分区再均衡,
    正常集群变动不再考虑范围内:

    1. 防止 因为未能及时发送心跳,导致Consumer 超时被踢出消费者组。
      这里可以设置 session.timeout.ms超时时间 和 heartbeat.interval.ms 心跳间隔
      一般可以把 超时时间设置为 心跳间隔的 3倍。
    2. Consumer消费时间过长导致的。
      Consumer端如果无法在规定时间内消费完 poll 来的消息,
      那么就认为该消费者有问题,从而该消费者会自主离组,
      所以我们可以设置 max.poll.interval.ms比处理时间略长。
    3. 从第二点我们还可能引申一点就是,如果集群经常发生 分区在均衡,
      那么你可能需要去观察下消费者执行任务的耗时,
      特别注意观察下 GC 的占用时间
  2. 分区分配过程
    上面我们提到了为 分区分配消费者,
    那么我们现在就来看看分配过程是怎么样的。

    1. 确定 群组协调器
      每当我们创建一个消费组,
      kafka 会为我们分配一个 broker 作为该消费组的 coordinator(协调器)

    2. 注册消费者 并选出 leader consumer
      当我们的有了 coordinator 之后,
      消费者将会开始往该 coordinator上进行注册,
      第一个注册的 消费者将成为该消费组的 leader,
      后续的 作为 follower

    3. 当 leader 选出来后,
      他会从coordinator那里实时获取分区 和 consumer 信息,
      并根据分区策略给每个consumer 分配 分区,
      并将分配结果告诉 coordinator。

    4. follower 消费者将从 coordinator 那里获取到自己相关的分区信息进行消费,
      对于所有的 follower 消费者而言,
      他们只知道自己消费的分区,
      并不知道其他消费者的存在。

    5. 至此,消费者都知道自己的消费的分区,
      分区过程结束,
      当发生 分区再均衡 的时候,
      leader 将会重复分配过程

实践——kafka 消费者的使用

咱们以 java api 为例,下面是一个简单的 kafka consumer

    public static void main(String[] args) {
        //consumer 的配置属性
        Properties props = new Properties();

        ///brokers 地址
        props.put("bootstrap.servers", "localhost:9092");

        //指定该 consumer 将加入的消费组
        props.put("group.id", "test");
        // 开启自动提交 offset,关于offset提交,我们后续再来详细说说
        props.put("enable.auto.commit", "true");
        props.put("auto.commit.interval.ms", "1000");

        //指定序列化类
        props.put("key.deserializer", "org.apache.kafka.common.serialization.StringDeserializer");
        props.put("value.deserializer", "org.apache.kafka.common.serialization.StringDeserializer");
        
        //创建 consumer
        KafkaConsumer<String, String> consumer = new KafkaConsumer<>(props);
        
        //订阅消费主题,这里一个消费者可以同时消费 foo 和 bar 两个主题的数据
        consumer.subscribe(Arrays.asList("foo", "bar"));
        while (true) {
            ConsumerRecords<String, String> records = consumer.poll(100);
            for (ConsumerRecord<String, String> record : records)
                System.out.printf("offset = %d, key = %s, value = %s%n", record.offset(), record.key(), record.value());
        }
    }

使用起来还是很简单的,不过如果想要用好 consumer,
可能你还需要了解以下这些东西:

  1. 分区控制策略
  2. consumer 的一些常用配置
  3. offset 的控制

ok,那么我们接下来一个个来看吧。。。

分区控制策略

  1. 手动控制分区
    咱们先来说下最简单的手动分区控制,代码如下:
     String topic = "foo";
     TopicPartition partition0 = new TopicPartition(topic, 0);
     TopicPartition partition1 = new TopicPartition(topic, 1);
     consumer.assign(Arrays.asList(partition0, partition1));

看起来只是把普通的订阅方式修改成了订阅 topic 指定的分区,
其余的还是照常使用,不过这里也需要注意一下的是:

  • 一般只作为独立消费者,
    也就是不能加入消费组,
    或者说他本身就是作为一个消费组存在,
    要保证这一点,我们只需要保证其group id 是唯一的就可以了。

  • 对于topic的分区变动不敏感,
    也就是说当 topic新增了分区,
    分区的数据将会发生改变,
    但该消费组对此确是不感知的,依然照常运行,
    所以很多时候需要你手动consumer.partitionsFor()去查看topic的分区情况

  • 不要和 subscription混合使用

  1. 使用partition.assignment.strategy进行分区策略配置
    这里的话 kafka 是自带两种分区策略的,
    为了方便理解,
    我们以如下场景为例来进行解释:
    已知:
    TopicA 有 3 个 partition(分区):A-1,A-2,A-3;
    TopicB 有 3 个 partition(分区):B-1,B-2,B-3;
    ConsumerA 和 ConsumerB 作为一个消费组 ConsumerGroup 同时消费 TopicA 和 TopicB
  • Range
    该方式最大的特点就是会将连续的分区分配给一个消费者,
    根据示例,我们可以得出如下结论:

    ConsumerGroup 消费 TopicA 的时候:
    ConsumerA 会分配到 A-1,A-2
    ConsumerB 会分配到 A-3

    ConsumerGroup 消费 TopicB 的时候:
    ConsumerA 会分配到 B-1,B-2
    ConsumerB 会分配到 B-3

    所以:
    ConsumerA 分配到了4个分区: A-1,A-2,B-1,B-2
    ConsumerB 分配到了2个分区:A-3,B-3

  • RoundRobin
    该方式最大的特点就是会以轮询的方式将分区分配给一个个消费者,
    根据示例,我们可以得出如下结论:

    ConsumerGroup 消费 TopicA 的时候:
    ConsumerA 分配到 A-1
    ConsumerB 分配到 A-2
    ConsumerA 分配到 A-3

    ConsumerGroup 消费 TopicB 的时候,
    因为上次分配到了 ConsumerA,
    那么这次轮到 ConsumerB了 所以:

    ConsumerB 分配到 B-1
    ConsumerA 分配到 B-2
    ConsumerB 分配到 B-3

    所以:
    ConsumerA 分配到了4个分区: A-1,A-3,B-2
    ConsumerB 分配到了2个分区:A-2,B-1,B-3

从上面我们也是可以看出这两种策略的异同,
RoundRobin 相比较 Range 会使得分区分配的更加的均衡,
也是经常使用的一种方式,
Range则重在对于消费者消费的数据和 分区 关联上了,
在一些特殊的场景可能会用到

  • StickyAssignor
    这是Kafka 0.11.x 引入的一种分区策略,
    该分区算法的复杂度比较高,
    笔者目前没有研究,参考了一些博客和书籍,
    整理如下:
    该方式的特点:

    • 分区的分配要尽可能均匀。
    • 分区的分配尽可能与上次分配的保持相同。
    • 当两者发生冲突时,第一个目标优先于第二个目标

    之前的两种策略在再均衡的时候都没有考虑上次的分区的分配情况,
    该策略则对这一方面进行了补足,
    如果你之前使用的是 RoundRobin 这种方式,
    不妨换成 StickyAssignor 来试试。

  • 自定义的分区策略
    上面三种分区策略是 kafka 默认自带的策略,
    虽然大多数情况下够用了,
    但是可能针对一些特殊需求,
    我们也可以定义自己的分区策略

    1. Range分区策略源码
      如何自定义呢?
      最好的方式莫过于看源码是怎么实现的,
      然后自己依葫芦画瓢来一个,
      所以我们先来看看 Range分区策略源码,
      如下,我只做了简单的注释,因为它本身也很简单,
      重点看下 assign 的参数以及返回注释就 ok了
    public class RangeAssignor extends AbstractPartitionAssignor{
      //省略部分代码。。。。
     /**
       * 根据订阅者 和 分区数量来进行分区
       * @param partitionsPerTopic: topic->分区数量
       * @param subscriptions: memberId 消费者id -> subscription 消费者信息
       * @return: memberId ->list<topic名称 和 分区序号(id)>
       */
      @Override
      public Map<String, List<TopicPartition>> assign(Map<String, Integer> partitionsPerTopic, Map<String, Subscription> subscriptions) {
          //topic -> list<消费者>
          Map<String, List<String>> consumersPerTopic = consumersPerTopic(subscriptions);
    
          //初始化 返回结果
          Map<String, List<TopicPartition>> assignment = new HashMap<>();
          for (String memberId : subscriptions.keySet())
              assignment.put(memberId, new ArrayList<TopicPartition>());
    
          for (Map.Entry<String, List<String>> topicEntry : consumersPerTopic.entrySet()) {
              //topic
              String topic = topicEntry.getKey();
              // 消费该topic的 consumer-id
              List<String> consumersForTopic = topicEntry.getValue();
    
              //topic 的分区数量
              Integer numPartitionsForTopic = partitionsPerTopic.get(topic);
              if (numPartitionsForTopic == null)
                  continue;
    
              Collections.sort(consumersForTopic);
    
              //平均每个消费者分配的 分区数量
              int numPartitionsPerConsumer = numPartitionsForTopic / consumersForTopic.size();
              //平均之后剩下的 分区数
              int consumersWithExtraPartition = numPartitionsForTopic % consumersForTopic.size();
    
              //这里就是将连续分区切开然后分配给每个消费者
              List<TopicPartition> partitions = AbstractPartitionAssignor.partitions(topic, numPartitionsForTopic);
              for (int i = 0, n = consumersForTopic.size(); i < n; i++) {
                  int start = numPartitionsPerConsumer * i + Math.min(i, consumersWithExtraPartition);
                  int length = numPartitionsPerConsumer + (i + 1 > consumersWithExtraPartition ? 0 : 1);
                  assignment.get(consumersForTopic.get(i)).addAll(partitions.subList(start, start + length));
              }
          }
          return assignment;
        }
      }
    
    1. 自定义一个 分区策略
      这里先缓缓把,太简单把,没什么用,太复杂把,一时也想不出好的场景,
      如果你有需求,欢迎留言,我们一起来实现

Consumer 常用配置

首先,我们都应该知道,最全最全的文档应该是来自官网(虽然有时候可能官网找不到):
http://kafka.apachecn.org/documentation.html#newconsumerconfigs
嗯,以下内容来自 kafka权威指南 ,
请原谅我的小懒惰。。。后续有时间会把工作中的遇到的补充上

  1. fetch.min.bytes
    该属性指定了消费者从服务器获取记录的最小字节数。
    broker 在收到消费者的数据请求时,
    如果可用的数据量小于fetch.min.bytes 指定的大小,
    那么它会等到有足够的可用数据时才把它返回给消费者。
    这样可以降低消费者和 broker 的工作负载,
    因为它们在主题不是很活跃的时候(或者一天里的低谷时段),
    就不需要来来回回地处理消息。
    如果没有很多可用数据,但消费者的 CPU 使用率却很高,
    那么就需要把该属性的值设得比默认值大。
    如果消费者的数量比较多,
    把该属性的值设置得大一点可以降低broker 的工作负载。

  2. fetch.max.wait.ms
    我们通过 fetch.min.bytes 告诉 Kafka,
    等到有足够的数据时才把它返回给消费者。
    feth.max.wait.ms 则用于指定 broker 的等待时间,默认是 500ms。
    如果没有足够的数据流入 Kafka,
    消费者获取最小数据量的要求就得不到满足,
    最终导致 500ms 的延迟。
    如果要降低潜在的延迟(为了满足 SLA),
    可以把该参数值设置得小一些。
    如果 fetch.max.wait.ms 被设为 100ms,
    并且fetch.min.bytes 被设为 1MB,
    那么 Kafka 在收到消费者的请求后,
    要么返回 1MB 数据,
    要么在100ms 后返回所有可用的数据,
    就看哪个条件先得到满足。

  3. max.partition.fetch.bytes
    该属性指定了服务器从每个分区里返回给消费者的最大字节数。它的默认值是 1MB,也就是说,KafkaConsumer.poll() 方法从每个分区里返回的记录最多不超过 max.partition.fetch.bytes指定的字节。如果一个主题有 20 个分区和 5 个消费者,那么每个消费者需要至少 4MB 的可用内存来接收记录。在为消费者分配内存时,可以给它们多分配一些,因为如果群组里有消费者发生崩溃,剩下的消费者需要处理更多的分区。
    max.partition.fetch.bytes 的值必须比 broker 能够接收的最大消息的字节数(通过 max.message.size 属性配置)大,否则消费者可能无法读取这些消息,导致消费者一直挂起重试。在设置该属性时,另一个需要考虑的因素是消费者处理数据的时间。消费者需要频繁调用poll() 方法来避免会话过期和发生分区再均衡,如果单次调用 poll() 返回的数据太多,消费者需要更多的时间来处理,可能无法及时进行下一个轮询来避免会话过期。
    如果出现这种情况,可以把max.partition.fetch.bytes 值改小,或者延长会话过期时间。

  4. session.timeout.ms
    该属性指定了消费者在被认为死亡之前可以与服务器断开连接的时间,
    默认是 3s。
    如果消费者没有在session.timeout.ms 指定的时间内发送心跳给群组协调器,
    就被认为已经死亡,
    协调器就会触发再均衡,
    把它的分区分配给群组里的其他消费者。
    该属性与 heartbeat.interval.ms 紧密相关。
    heartbeat.interval.ms 指定了 poll() 方法向协调器发送心跳的频率,
    session.timeout.ms 则指定了消费者可以多久不发送心跳。
    所以,一般需要同时修改这两个属性,
    heartbeat.interval.ms 必须比 session.timeout.ms 小,
    一般是session.timeout.ms 的三分之一。
    如果 session.timeout.ms 是 3s,那么 heartbeat.interval.ms 应该是 1s。
    把session.timeout.ms 值设得比默认值小,
    可以更快地检测和恢复崩溃的节点,
    不过长时间的轮询或垃圾收集可能导致非预期的再均衡。
    把该属性的值设置得大一些,
    可以减少意外的再均衡,
    不过检测节点崩溃需要更长的时间。

  5. auto.offset.reset
    该属性指定了消费者在读取一个没有偏移量的分区或者偏移量无效的情况下
    (因消费者长时间失效,包含偏移量的记录已经过时并被删除)该作何处理。
    它的默认值是 latest,
    意思是说,
    在偏移量无效的情况下,
    消费者将从最新的记录开始读取数据(在消费者启动之后生成的记录)。
    另一个值是earliest,
    意思是说,
    在偏移量无效的情况下,
    消费者将从起始位置读取分区的记录。

  6. enable.auto.commit
    我们稍后将介绍几种不同的提交偏移量的方式。
    该属性指定了消费者是否自动提交偏移量,默认值是true。
    为了尽量避免出现重复数据和数据丢失,可以把它设为 false,
    由自己控制何时提交偏移量。
    如果把它设为true,还可以通过配置 auto.commit.interval.ms 属性来控制提交的频率。

  7. partition.assignment.strategy(这部分好像重复了 ~~~)
    我们知道,分区会被分配给群组里的消费者。
    PartitionAssignor 根据给定的消费者和主题,
    决定哪些分区应该被分配给哪个消费者。
    Kafka 有两个默认的分配策略。

    • Range
        该策略会把主题的若干个连续的分区分配给消费者。假设消费者 C1 和消费者 C2 同时订阅了主题T1 和 主题 T2,并且每个主题有 3 个分区。那么消费者 C1 有可能分配到这两个主题的分区 0 和分区 1,而消费者 C2 分配到这两个主题的分区 2。因为每个主题拥有奇数个分区,而分配是在主题内独立完成的,第一个消费者最后分配到比第二个消费者更多的分区。只要使用了 Range 策略,而且分区数量无法被消费者数量整除,就会出现这种情况。

    • RoundRobin
        该策略把主题的所有分区逐个分配给消费者。如果使用 RoundRobin 策略来给消费者 C1 和消费者C2 分配分区,那么消费者 C1 将分到主题 T1 的分区 0 和分区 2 以及主题 T2 的分区 1,消费者 C2 将分配到主题 T1 的分区 1 以及主题 T2 的分区 0 和分区 2。一般来说,如果所有消费者都订阅相同的主题(这种情况很常见),RoundRobin 策略会给所有消费者分配相同数量的分区(或最多就差一个分区)。可以通过设置 partition.assignment.strategy 来选择分区策略。
      默认使用的是org.apache.kafka.clients.consumer.RangeAssignor,这个类实现了 Range 策略,不过也可以把它改成 org.apache.kafka.clients.consumer.RoundRobinAssignor。我们还可以使用自定义策略,在这种情况下,partition.assignment.strategy 属性的值就是自定义类的名字。

  8. client.id
    该属性可以是任意字符串,
    broker 用它来标识从客户端发送过来的消息,
    通常被用在日志、度量指标和配额里。

  9. max.poll.records
    该属性用于控制单次调用 call() 方法能够返回的记录数量,
    可以帮你控制在轮询里需要处理的数据量。

  10. receive.buffer.bytes 和 send.buffer.bytes
    socket 在读写数据时用到的 TCP 缓冲区也可以设置大小。
    如果它们被设为 -1,就使用操作系统的默认值。
    如果生产者或消费者与 broker 处于不同的数据中心内,
    可以适当增大这些值,
    因为跨数据中心的网络一般都有比较高的延迟和比较低的带宽

offset 的控制

这篇文章有点太长了,所以准备另起一篇来专门讲 offset 的控制。
预计在周末更新吧,如果你有兴趣,可以点击关注一下,以便及时收到提醒噢!!!
弱弱的,也是求一波关注,哈哈哈!!!

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

推荐阅读更多精彩内容

  • 姓名:周小蓬 16019110037 转载自:http://blog.csdn.net/YChenFeng/art...
    aeytifiw阅读 34,719评论 13 425
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,644评论 18 139
  • kafka的定义:是一个分布式消息系统,由LinkedIn使用Scala编写,用作LinkedIn的活动流(Act...
    时待吾阅读 5,314评论 1 15
  • 大致可以通过上述情况进行排除 1.kafka服务器问题 查看日志是否有报错,网络访问问题等。 2. kafka p...
    生活的探路者阅读 7,584评论 0 10
  • 茶余饭后 作者老兵 像涓涓细流, 从靜静的泉眼涌出, 一路吮吸着大地的乳汁, 伴着花香、小草、乌儿, 潺潺小溪,快...
    L1老兵李培义阅读 331评论 1 2