Kafka 多维度系统精讲(10)- kafka consumer(4)

1 consumer 控制 offset 起始位置

 /**
     * 手动指定 offset 的起始位置,及手动提交 offset
     */
    private static void controlOffset() {
        Properties props = new Properties();
        props.setProperty("bootstrap.servers", "192.168.10.103:9092");
        props.setProperty("group.id", "test");
        props.setProperty("enable.auto.commit", "true");
        props.setProperty("auto.commit.interval.ms", "1000");
        props.setProperty("key.deserializer", "org.apache.kafka.common.serialization.StringDeserializer");
        props.setProperty("value.deserializer", "org.apache.kafka.common.serialization.StringDeserializer");

        KafkaConsumer<String, String> consumer = new KafkaConsumer<>(props);

        // tzb-new-topic, 0,1 两个 partition
        TopicPartition p0 = new TopicPartition(TOPIC_NAME, 0);

        // 订阅一个或者多个 topic
//        consumer.subscribe(Arrays.asList(TOPIC_NAME));

        // 消费订阅某个 topic 的某个分区
        consumer.assign(Arrays.asList(p0));

        while (true) {

            // 手动指定 offset 起始位置
            /**
             * 1、人为控制 offset 起始位置
             * 2、如果出现程序错误,重复消费
             */
            /**
             * 1.第一次从 0 消费【一般情况】
             * 2.比如一次消费了100条, offset 置为101,并且存入 redis
             * 3.每次 poll 之前,从redis中获取最新的 offset 位置
             * 4.每次从这个位置开始消费
             */
            consumer.seek(p0, 400);

            // 定时间隔去拉取
            ConsumerRecords<String, String> records = consumer.poll(Duration.ofMillis(1000));

            // 每个 partition 单独处理
            for (TopicPartition partition : records.partitions()) {
                List<ConsumerRecord<String, String>> pRecord = records.records(partition);
                for (ConsumerRecord<String, String> record : pRecord) {
                    System.out.printf("partition = %d, offset = %d, key = %s, value = %s%n",
                            record.partition(), record.offset(), record.key(), record.value());
                }

                long lastOffset = pRecord.get(pRecord.size() - 1).offset();

                // 单个 partition 中的 offset,并且进行提交
                Map<TopicPartition, OffsetAndMetadata> offset = new HashMap<>();
                offset.put(partition, new OffsetAndMetadata(lastOffset + 1));
                // 提交 offset
                consumer.commitSync(offset);
                System.out.println("===========  partition - " + partition +"====================");
            }

        }
    }

2 Consumer 限流

 /**
     * 流量控制:限流
     */
    private static void controlPause() {
        Properties props = new Properties();
        props.setProperty("bootstrap.servers", "192.168.10.103:9092");
        props.setProperty("group.id", "test");
        props.setProperty("enable.auto.commit", "true");
        props.setProperty("auto.commit.interval.ms", "1000");
        props.setProperty("key.deserializer", "org.apache.kafka.common.serialization.StringDeserializer");
        props.setProperty("value.deserializer", "org.apache.kafka.common.serialization.StringDeserializer");

        KafkaConsumer<String, String> consumer = new KafkaConsumer<>(props);

        // tzb-new-topic, 0,1 两个 partition
        TopicPartition p0 = new TopicPartition(TOPIC_NAME, 0);
        TopicPartition p1 = new TopicPartition(TOPIC_NAME, 1);

        // 消费订阅某个 topic 的某个分区
        consumer.assign(Arrays.asList(p0, p1));

        long totalNum = 40;


        while (true) {
            // 定时间隔去拉取
            ConsumerRecords<String, String> records = consumer.poll(Duration.ofMillis(1000));

            // 每个 partition 单独处理
            for (TopicPartition partition : records.partitions()) {
                List<ConsumerRecord<String, String>> pRecord = records.records(partition);

                long num = 0;

                for (ConsumerRecord<String, String> record : pRecord) {
                    System.out.printf("partition = %d, offset = %d, key = %s, value = %s%n",
                            record.partition(), record.offset(), record.key(), record.value());

                    /*
                     * 1.接收到record 信息后,去令牌桶中拿取令牌
                     * 2.如果获取到令牌,则继续业务处理
                     * 3.如果获取不到令牌,则 pause 等待令牌
                     * 4.当令牌桶中的令牌足够,则将 consumer 置为 resume 状态
                     */
                    num++;
                    if (record.partition() == 0) {
                        if (num >= totalNum) {
                            consumer.pause(Arrays.asList(p0));
                        }
                    }

                    if(record.partition() == 1){
                        if(num == 40){
                            consumer.resume(Arrays.asList(p0));
                        }
                    }

                }

                long lastOffset = pRecord.get(pRecord.size() - 1).offset();

                // 单个 partition 中的 offset,并且进行提交
                Map<TopicPartition, OffsetAndMetadata> offset = new HashMap<>();
                offset.put(partition, new OffsetAndMetadata(lastOffset + 1));
                // 提交 offset
                consumer.commitSync(offset);
                System.out.println("===========  partition - " + partition + "====================");
            }

        }
    }

3 Consumer Rebalance

在这里插入图片描述

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值