1. 程式人生 > >Kafka 環境部署搭建

Kafka 環境部署搭建

kafka 環境部署搭建

一、基本概念

介紹

Kafka是一個分布式的、可分區的、可復制的消息系統。它提供了普通消息系統的功能,但具有自己獨特的設計。

這個獨特的設計是什麽樣的呢?

首先讓我們看幾個基本的消息系統術語:
Kafka將消息以topic為單位進行歸納。
將向Kafka topic發布消息的程序成為producers.
將預訂topics並消費消息的程序成為consumer.
Kafka以集群的方式運行,可以由一個或多個服務組成,每個服務叫做一個broker.
producers通過網絡將消息發送到Kafka集群,集群向消費者提供消息,如下圖所示:

技術分享



客戶端和服務端通過TCP協議通信。Kafka提供了Java客戶端,並且對多種語言都提供了支持。


Topics 和Logs

先來看一下Kafka提供的一個抽象概念:topic.
一個topic是對一組消息的歸納。對每個topic,Kafka 對它的日誌進行了分區,如下圖所示:
技術分享


每個分區都由一系列有序的、不可變的消息組成,這些消息被連續的追加到分區中。分區中的每個消息都有一個連續的序列號叫做offset,用來在分區中唯一的標識這個消息。
在一個可配置的時間段內,Kafka集群保留所有發布的消息,不管這些消息有沒有被消費。比如,如果消息的保存策略被設置為2天,那麽在一個消息被發布的兩天時間內,它都是可以被消費的。之後它將被丟棄以釋放空間。Kafka的性能是和數據量無關的常量級的,所以保留太多的數據並不是問題。

實際上每個consumer唯一需要維護的數據是消息在日誌中的位置,也就是offset.這個offset有consumer來維護:一般情況下隨著consumer不斷的讀取消息,這offset的值不斷增加,但其實consumer可以以任意的順序讀取消息,比如它可以將offset設置成為一個舊的值來重讀之前的消息。


以上特點的結合,使Kafka consumers非常的輕量級:它們可以在不對集群和其他consumer造成影響的情況下讀取消息。你可以使用命令行來"tail"消息而不會對其他正在消費消息的consumer造成影響。

將日誌分區可以達到以下目的:首先這使得每個日誌的數量不會太大,可以在單個服務上保存。另外每個分區可以單獨發布和消費,為並發操作topic提供了一種可能。

分布式

每個分區在Kafka集群的若幹服務中都有副本,這樣這些持有副本的服務可以共同處理數據和請求,副本數量是可以配置的。副本使Kafka具備了容錯能力。
每個分區都由一個服務器作為“leader”,零或若幹服務器作為“followers”,leader負責處理消息的讀和寫,followers則去復制leader.如果leader down了,followers中的一臺則會自動成為leader。集群中的每個服務都會同時扮演兩個角色:作為它所持有的一部分分區的leader,同時作為其他分區的followers,這樣集群就會據有較好的負載均衡。

Producers

Producer將消息發布到它指定的topic中,並負責決定發布到哪個分區。通常簡單的由負載均衡機制隨機選擇分區,但也可以通過特定的分區函數選擇分區。使用的更多的是第二種。


Consumers

發布消息通常有兩種模式:隊列模式(queuing)和發布-訂閱模式(publish-subscribe)。隊列模式中,consumers可以同時從服務端讀取消息,每個消息只被其中一個consumer讀到;發布-訂閱模式中消息被廣播到所有的consumer中。Consumers可以加入一個consumer 組,共同競爭一個topic,topic中的消息將被分發到組中的一個成員中。同一組中的consumer可以在不同的程序中,也可以在不同的機器上。如果所有的consumer都在一個組中,這就成為了傳統的隊列模式,在各consumer中實現負載均衡。如果所有的consumer都不在不同的組中,這就成為了發布-訂閱模式,所有的消息都被分發到所有的consumer中。更常見的是,每個topic都有若幹數量的consumer組,每個組都是一個邏輯上的“訂閱者”,為了容錯和更好的穩定性,每個組由若幹consumer組成。這其實就是一個發布-訂閱模式,只不過訂閱者是個組而不是單個consumer。

技術分享

由兩個機器組成的集群擁有4個分區 (P0-P3) 2個consumer組. A組有兩個consumerB組有4個

相比傳統的消息系統,Kafka可以很好的保證有序性。
傳統的隊列在服務器上保存有序的消息,如果多個consumers同時從這個服務器消費消息,服務器就會以消息存儲的順序向consumer分發消息。雖然服務器按順序發布消息,但是消息是被異步的分發到各consumer上,所以當消息到達時可能已經失去了原來的順序,這意味著並發消費將導致順序錯亂。為了避免故障,這樣的消息系統通常使用“專用consumer”的概念,其實就是只允許一個消費者消費消息,當然這就意味著失去了並發性。

在這方面Kafka做的更好,通過分區的概念,Kafka可以在多個consumer組並發的情況下提供較好的有序性和負載均衡。將每個分區分只分發給一個consumer組,這樣一個分區就只被這個組的一個consumer消費,就可以順序的消費這個分區的消息。因為有多個分區,依然可以在多個consumer組之間進行負載均衡。註意consumer組的數量不能多於分區的數量,也就是有多少分區就允許多少並發消費。

Kafka只能保證一個分區之內消息的有序性,在不同的分區之間是不可以的,這已經可以滿足大部分應用的需求。如果需要topic中所有消息的有序性,那就只能讓這個topic只有一個分區,當然也就只有一個consumer組消費它。

###########################################
二、環境搭建


Step 1: 下載Kafka

點擊下載最新的版本並解壓.

  1. > tar -xzf kafka_2.9.2-0.8.1.1.tgz

  2. > cd kafka_2.9.2-0.8.1.1

復制代碼




Step 2: 啟動服務

Kafka用到了Zookeeper,所有首先啟動Zookper,下面簡單的啟用一個單實例的Zookkeeper服務。可以在命令的結尾加個&符號,這樣就可以啟動後離開控制臺。

  1. > bin/zookeeper-server-start.sh config/zookeeper.properties &

  2. [2013-04-22 15:01:37,495] INFO Reading configuration from: config/zookeeper.properties (org.apache.zookeeper.server.quorum.QuorumPeerConfig)

  3. ...

復制代碼



現在啟動Kafka:

  1. > bin/kafka-server-start.sh config/server.properties

  2. [2013-04-22 15:01:47,028] INFO Verifying properties (kafka.utils.VerifiableProperties)

  3. [2013-04-22 15:01:47,051] INFO Property socket.send.buffer.bytes is overridden to 1048576 (kafka.utils.VerifiableProperties)

  4. ...

復制代碼



Step 3: 創建 topic

創建一個叫做“test”的topic,它只有一個分區,一個副本。

  1. > bin/kafka-topics.sh --create --zookeeper localhost:2181 --replication-factor 1 --partitions 1 --topic test

復制代碼



可以通過list命令查看創建的topic:

  1. > bin/kafka-topics.sh --list --zookeeper localhost:2181

  2. test

復制代碼



除了手動創建topic,還可以配置broker讓它自動創建topic.

Step 4:發送消息.

Kafka 使用一個簡單的命令行producer,從文件中或者從標準輸入中讀取消息並發送到服務端。默認的每條命令將發送一條消息。

運行producer並在控制臺中輸一些消息,這些消息將被發送到服務端:

  1. > bin/kafka-console-producer.sh --broker-list localhost:9092 --topic test

  2. This is a messageThis is another message

復制代碼



ctrl+c可以退出發送。

Step 5: 啟動consumer

Kafka also has a command line consumer that will dump out messages to standard output.
Kafka也有一個命令行consumer可以讀取消息並輸出到標準輸出:

  1. > bin/kafka-console-consumer.sh --zookeeper localhost:2181 --topic test --from-beginning

  2. This is a message

  3. This is another message

復制代碼



你在一個終端中運行consumer命令行,另一個終端中運行producer命令行,就可以在一個終端輸入消息,另一個終端讀取消息。
這兩個命令都有自己的可選參數,可以在運行的時候不加任何參數可以看到幫助信息。

Step 6: 搭建一個多個broker的集群

剛才只是啟動了單個broker,現在啟動有3個broker組成的集群,這些broker節點也都是在本機上的:
首先為每個節點編寫配置文件:

  1. > cp config/server.properties config/server-1.properties

  2. > cp config/server.properties config/server-2.properties

復制代碼



在拷貝出的新文件中添加以下參數:

  1. config/server-1.properties:

  2. broker.id=1

  3. port=9093

  4. log.dir=/tmp/kafka-logs-1

復制代碼



  1. config/server-2.properties:

  2. broker.id=2

  3. port=9094

  4. log.dir=/tmp/kafka-logs-2

復制代碼



broker.id在集群中唯一的標註一個節點,因為在同一個機器上,所以必須制定不同的端口和日誌文件,避免數據被覆蓋。

We already have Zookeeper and our single node started, so we just need to start the two new nodes:
剛才已經啟動可Zookeeper和一個節點,現在啟動另外兩個節點:

  1. > bin/kafka-server-start.sh config/server-1.properties &

  2. ...

  3. > bin/kafka-server-start.sh config/server-2.properties &

  4. ...

復制代碼



創建一個擁有3個副本的topic:

  1. > bin/kafka-topics.sh --create --zookeeper localhost:2181 --replication-factor 3 --partitions 1 --topic my-replicated-topic

復制代碼



現在我們搭建了一個集群,怎麽知道每個節點的信息呢?運行“"describe topics”命令就可以了:

  1. > bin/kafka-topics.sh --describe --zookeeper localhost:2181 --topic my-replicated-topic

復制代碼


  1. Topic:my-replicated-topic PartitionCount:1 ReplicationFactor:3 Configs:

  2. Topic: my-replicated-topic Partition: 0 Leader: 1 Replicas: 1,2,0 Isr: 1,2,0

復制代碼



下面解釋一下這些輸出。第一行是對所有分區的一個描述,然後每個分區都會對應一行,因為我們只有一個分區所以下面就只加了一行。
leader:負責處理消息的讀和寫,leader是從所有節點中隨機選擇的.
replicas:列出了所有的副本節點,不管節點是否在服務中.
isr:是正在服務中的節點.
在我們的例子中,節點1是作為leader運行。
向topic發送消息:

  1. > bin/kafka-console-producer.sh --broker-list localhost:9092 --topic my-replicated-topic

復制代碼


  1. ...

  2. my test message 1my test message 2^C

復制代碼



消費這些消息:

  1. > bin/kafka-console-consumer.sh --zookeeper localhost:2181 --from-beginning --topic my-replicated-topic

復制代碼


  1. ...

  2. my test message 1

  3. my test message 2

  4. ^C

復制代碼



測試一下容錯能力.Broker 1作為leader運行,現在我們kill掉它:

  1. > ps | grep server-1.properties7564 ttys002 0:15.91 /System/Library/Frameworks/JavaVM.framework/Versions/1.6/Home/bin/java...

  2. > kill -9 7564

復制代碼



另外一個節點被選做了leader,node 1 不再出現在 in-sync 副本列表中:

  1. > bin/kafka-topics.sh --describe --zookeeper localhost:218192 --topic my-replicated-topic

  2. Topic:my-replicated-topic PartitionCount:1 ReplicationFactor:3 Configs:

  3. Topic: my-replicated-topic Partition: 0 Leader: 2 Replicas: 1,2,0 Isr: 2,0

復制代碼



雖然最初負責續寫消息的leader down掉了,但之前的消息還是可以消費的:

  1. > bin/kafka-console-consumer.sh --zookeeper localhost:2181 --from-beginning --topic my-replicated-topic

  2. ...

  3. my test message 1

  4. my test message 2

復制代碼




看來Kafka的容錯機制還是不錯的。

################################################


本文出自 “李世龍” 博客,謝絕轉載!

Kafka 環境部署搭建