Producer拦截器(interceptor)是在Kafka 0.10版本被引入的,主要用于实现clients端的定制化控制逻辑。
对于producer而言,interceptor使得用户在消息发送前以及producer回调逻辑前有机会对消息做一些定制化需求,比如修改消息等。同时,producer允许用户指定多个interceptor按序作用于同一条消息从而形成一个拦截链(interceptor chain)。
直接上demo
import java.util.Map;
import org.apache.kafka.clients.producer.ProducerInterceptor;
import org.apache.kafka.clients.producer.ProducerRecord;
import org.apache.kafka.clients.producer.RecordMetadata;
public class TimeInterceptor implements ProducerInterceptor<String, String> {
@Override
//获取配置信息和初始化数据时调用。
public void configure(Map<String, ?> configs) {
}
@Override
//该方法封装进KafkaProducer.send方法中,即它运行在用户主线程中
//Producer确保在消息被序列化以及计算分区前调用该方法
//用户可以在该方法中对消息做任何操作
//但最好保证不要修改消息所属的topic和分区,否则会影响目标分区的计算。
public ProducerRecord<String, String> onSend(ProducerRecord<String, String> record) {
// 创建一个新的record,把时间戳写入消息体的最前部
return new ProducerRecord(record.topic(), record.partition(), record.timestamp(), record.key(),
System.currentTimeMillis() + "," + record.value().toString());
}
@Override
//该方法会在消息从RecordAccumulator成功发送到Kafka Broker之后,或者在发送过程中失败时调用。
//并且通常都是在producer回调逻辑触发之前
//onAcknowledgement运行在producer的IO线程中
//因此不要在该方法中放入很重的逻辑,否则会拖慢producer的消息发送效率。
public void onAcknowledgement(RecordMetadata metadata, Exception exception) {
// 统计成功和失败的次数
if (exception == null) {
successCounter++;
} else {
errorCounter++;
}
}
@Override
//关闭interceptor,主要用于执行一些资源清理工作
public void close() {
}
}
在构建prodicer时候加上拦截器
List<String> interceptors = new ArrayList<>();
interceptors.add("com.atguigu.kafka.interceptor.TimeInterceptor"); interceptors.add("com.atguigu.kafka.interceptor.CounterInterceptor");
props.put(ProducerConfig.INTERCEPTOR_CLASSES_CONFIG, interceptors);
Producer<String, String> producer = new KafkaProducer<>(props);