本篇主要介绍如何进行producer的开发,为了进行相关测试,建议先按照本系列前两篇文章安装单机kafka或者kafka集群。
一、producer示例程序开发
首先引入kafka相关依赖,在pom.xml文件中加入如下依赖:
<!--kafka--> <dependency> <groupId>org.apache.kafka</groupId> <artifactId>kafka_2.12</artifactId> <version>2.2.0</version> </dependency>
在resources下面创建 kafka.properties 配置文件,用于设置kafka参数,内容如下:
bootstrap.servers=192.168.184.128:9092,192.168.184.128:9093,192.168.184.128:9094 key.serializer=org.apache.kafka.common.serialization.StringSerializer value.serializer=org.apache.kafka.common.serialization.StringSerializer acks=-1 retries=3 batch.size=323840 linger.ms=10 buffer.memory=33554432 max.block.ms=3000
其中,前三个参数必须明确指定,因为这三个参数没有默认值(注:kafka的producer参数配置可以参考http://kafka.apache.org/documentation/#producerconfigs),然后编写producer发送消息的代码:
/** * Kafka发送消息测试 * @throws IOException */ public void sendMsg() throws IOException { //1.构造properties对象 Properties properties = new Properties(); FileInputStream fileInputStream = new FileInputStream("F:\\javaCode\\jvmdemo\\src\\main\\resources\\kafka.properties"); properties.load(fileInputStream); fileInputStream.close(); //2.构造kafkaProducer对象 KafkaProducer producer = new KafkaProducer(properties); for (int i = 0; i < 100; i++) { //3.构造待发送消息的producerRecord对象,并指定消息要发送到哪个topic,消息的key和value ProducerRecord testTopic = new ProducerRecord("testTopic", Integer.toString(i), Integer.toString(i)); //4.调用kafkaProducer对象的send方法发送消息 producer.send(testTopic); } //5.关闭kafkaProducer producer.close(); }
然后登陆kafka所在服务器,执行以下命令监听消息:
cd /usr/local/kafka/bin ./kafka-console-consumer.sh --bootstrap-server 192.168.184.128:9092,192.168.184.128:9093,192.168.184.128:9094 --topic testTopic --from-beginning
运行sendMsg方法,注意观察消费端,
可以看到有0-99之间的数字依次被消费到,说明消息发送成功。
二、异步和同步发送消息
上面发送消息的示例程序中,在发完消息之后,没有对发送结果进行处理,如果消息发送失败我们也是无法得知的,这种方法在实际应用中是不推荐的。在实际使用场景中,一般使用同步和异步两种常见发送方式。Java版本producer的send方法会返回一个Future对象,如果调用Future.get()方法就会无限等待返回结果,实现同步发送的效果,否则就是异步发送。因此,同步的实现比较简单,只要调用Future.get()方法即可,对于异步发送,producer的send方法可以传入一个回调参数来实现对发送结果的响应,具体代码如下:
//4.调用kafkaProducer对象的send方法发送消息 Future send = producer.send(testTopic, new Callback() { @Override public void onCompletion(RecordMetadata recordMetadata, Exception exception) { if(null==exception){ //消息发送成功后的处理 System.out.println("消息发送成功"); }else{ //消息发送失败后的处理 System.out.println("消息发送失败"); } } }
以上代码中,send方法第二个参数传入一个匿名内部类对象,也可以传入实现了 org.apache.kafka.clients.producer.Callback 接口的类对象。同时 onCompletion 方法的两个入参 recordMetadata 和 exception 不会同时为空,当消息发送成功后, exception 为null,消息发送失败后 recordMetadata 为null。因此可以按照两个入参进行成功和失败逻辑的处理。
三、其他高级特性
1.消息分区机制
kafka producer提供了分区策略以及分区器(partitioner)用于确定将消息发送到指定topic的哪个分区中。默认分区器根据murmur2算法计算消息key的哈希值,然后对总分区数求模确认消息要被发送的目标分区号(这点让我想起了redis集群中key值的分配方法),这样就确保了相同key的消息被发送到相同分区。若消息没有key值,将采用轮询的方式确保消息在topic的所有分区上均匀分配。
除了使用kafka默认的分区机制,也可以通过实现 org.apache.kafka.clients.producer.Partitioner 接口来自定义分区器,此时需要在构造 KafkaProducer 的 properties 中增加 partitioner.class 来指明分区器实现类,如: partitioner.class=com.demo.service.CustomerPartitioner 。
2.消息序列化
在本篇开始的producer示例程序中,在构造 KafkaProducer 对象的时候,有两个配置项 key.serializer=org.apache.kafka.common.serialization.StringSerializer 和 value.serializer=org.apache.kafka.common.serialization.StringSerializer 分别用于配置消息key和value的序列化方式为String类型,除此之外,Kafka中还提供了如下默认的序列化器:
ByteArraySerializer :本质上什么也不做,因为网络中传输就是以字节传输的;
ByteBufferSerializer :序列化ByteBuffer消息;
BytesSerializer :序列化kafka自定义的Bytes类型;
IntegerSerializer :序列化Integer类型;
DoubleSerializer :序列化Double类型;
LongSerializer :序列化Long类型;
如果要自定义序列化器,则需要实现 org.apache.kafka.common.serialization.Serializer 接口,并且将 key.serializer 和 value.serializer 配置为自定义的序列化器。
3.消息压缩
消息压缩可以显著降低磁盘占用以及带宽占用,从而有效提升I/O密集型应用性能,但是引入压缩同时会消耗额外的CPU,因此压缩是I/O性能和CPU资源的平衡。kafka目前支持3种压缩算法:CZIP,Snappy和LZ4,性能测试结果显示三种压缩算法的性能如下:LZ4>>Snappy>GZIP,目前启用LZ4进行消息压缩的producer的吞吐量是最高的。
默认情况下Kafka是不压缩消息的,但是可以通过在创建 KafkaProducer 对象的时候设置producer端参数 compression.type 来开启消息压缩,如配置 compression.type=LZ4 。那么什么时候开启压缩呢?首先判断是否启用压缩的依据是I/O资源消耗与CPU资源消耗的对比,如果环境上I/O资源非常紧张,比如producer程序占用了大量的网络带宽或broker端的磁盘占用率很高,而producer端的CPU资源非常富裕,那么就可以考虑为producer开启压缩。
4.无消息丢失配置
在使用 KafkaProducer.send() 方法发送消息的时候,其实是把消息放入缓冲区中,再由一个专属I/O线程负责从缓冲区提取消息并封装消息到batch中,然后再发送出去。如果在I/O线程将消息发送出去之前,producer奔溃了,那么所有的消息都将丢失。同时,存在多消息发送时候由于网络抖动导致消息乱序的问题,为了解决这两个问题,可以通过在producer端以及broker端进行配置进行避免。
4.1 producer端配置
max.block.ms=3000 :设置block的时长,当缓冲区被填满或者metadata丢失时产生block,停止接收新的消息;
acks=all :等待所有follower都响应了发送消息认为消息发送成功;
retries=Integer.MAX_VALUE :设置重试次数,设置一个比较大的值可以保证消息不丢失;
max.in.flight.requests.per.connection=1 :限制producer在单个broker连接上能够发送的未响应请求的数量,从而防止同topic统一分区下消息乱序问题;
除了设置以上参数之外,在发送消息的时候,应该尽量使用带有回调参数的send方法来处理发送结果,如果数据发送失败,则显示调用 KafkaProducer.close(0) 方法来立即关闭producer,防止消息乱序。
4.2 broker端配置
unclean.leader.election.enable=false :关闭unclean leader选举,即不允许非ISR中的副本被选举为leader;
replication.factor>=3 :至少使用3个副本保存数据;
min.issync.replicas>1 :控制某条消息至少被写入到ISR中多少个副本才算成功,当且仅当producer端acks参数设置为all或者-1时,该参数才有效。
最后,确保 replication.factor>min.issync.replicas ,如果两者相等,那么只要有一个副本挂掉,分区就无法工作,推荐配置 replication.factor=min.issync.replicas+1 。
关于producer端的开发就介绍到这儿,下一篇将介绍consumer端的开发。
原文:https://www.cnblogs.com/fengweiweicoder/p/10848762.html