Zookeeper是一個分佈式協調服務,換言之,就是爲用戶的分佈式應用程序提供協調服務java
Zookeeper在配置文件中並無指定master和slave,啓動以後經過內部的選舉機制選舉出leader和follower,並且只有一個leader,其餘則爲follower。zookeeper集羣中只要有半數以上節點存活,集羣就能提供服務。 2.zookeeper集羣機制 半數機制:集羣中半數以上機器存活,集羣可用。 zookeeper適合裝在奇數臺機器上!!!node
tar -zxvf zookeeper-3.4.5.tar.gz
複製代碼
mv zookeeper-3.4.5 zookeeper(重命名文件夾zookeeper-3.4.5爲zookeeper)
複製代碼
vi /etc/profile
添加內容:
export ZOOKEEPER_HOME=/apps/package/zookeeper
export PATH=$PATH:$JAVA_HOME/bin:$ZOOKEEPER_HOME/bin
複製代碼
添加內容:
dataDir=/apps/package/zookeeper/data
dataLogDir=/apps/package/zookeeper/log
server.1=mini1:2888:3888
server.2=mini2:2888:3888
server.3=mini3:2888:3888
複製代碼
cd /apps/package/zookeeper
mkdir -m 755 data
mkdir -m 755 log
複製代碼
cd data
vi myid
添加內容:
1
複製代碼
mini2和mini3服務器的請修改爲2,3,未來會按這個myid選中出leader和follow。linux
scp -r /apps/package/zookeeper root@mini2:/apps/package/
scp -r /apps/package/zookeeper root@mini3:/apps/package/
複製代碼
若是在mini1中ping不通mini2和mini3,須要在hosts文件中配置mini2和mini3的ip地址apache
zkServer.sh start
zkServer.sh start-foreground(能夠看到啓動日誌)
複製代碼
jps(查看進程)
zkServer.sh status(查看集羣狀態,主從信息)
複製代碼
若是報端口占用,參考下面連接解決:http://blog.csdn.net/u014686180/article/details/51767863api
zkCli.sh -主機名(ip):2181
如:zkCli.sh -mini2:2181
複製代碼
ls /
複製代碼
create /zk "myData「 複製代碼
get /zk
複製代碼
-監聽這個節點的變化,當另一個客戶端改變/zk時,輸出監聽到的變化bash
get /zk watch
複製代碼
set /zk "zsl「 複製代碼
delete /zk
複製代碼
rmr /zk
複製代碼
參考文檔:http://www.cnblogs.com/likehua/tag/zookeeper/服務器
<dependency>
<groupId>org.apache.zookeeper</groupId>
<artifactId>zookeeper</artifactId>
<version>3.4.10</version>
<type>pom</type>
</dependency>
複製代碼
public class SimpleZkClient {
private static final String CONNECT_URL = "mini1:2181,mini2:2181,mini3:2181";
private static final int SESSION_TIME_OUT = 2000;
ZooKeeper zkCli = null;
@Before
public void init() throws Exception{
zkCli = new ZooKeeper(CONNECT_URL, SESSION_TIME_OUT, new Watcher() {
@Override
public void process(WatchedEvent event) {
System.out.println(event.getType()+"-----------"+event.getPath());
try{
zkCli.getChildren("/", true);
}catch (Exception e){
}
}
});
}
/**
* @Description 添加節點數據
* @Author 劉俊重
*/
@Test
public void create() throws Exception{
// 參數1:要建立的節點的路徑 參數2:節點大數據 參數3:節點的權限 參數4:節點的類型。上傳的數據能夠是任何類型,但都要轉成byte[]
String s = zkCli.create("/zk", "test".getBytes(), ZooDefs.Ids.OPEN_ACL_UNSAFE, CreateMode.PERSISTENT);
}
/**
* @Description 判斷節點是否存在
* @Author 劉俊重
*/
@Test
public void isExist() throws Exception{
Stat exists = zkCli.exists("/zk", false);
System.out.println(null==exists?"不存在":"存在");
}
/**
* @Description 獲取節點數據
* @Author 劉俊重
*/
@Test
public void getData() throws Exception{
byte[] data = zkCli.getData("/zk", false, null);
System.out.println("節點數據:"+new String(data));
}
/**
* @Description 遍歷節點數據
* @Author 劉俊重
*/
@Test
public void getChildren() throws Exception{
List<String> children = zkCli.getChildren("/", false);
for(String s : children){
System.out.println("節點名稱:"+s);
}
Thread.sleep(Long.MAX_VALUE);
}
/**
* @Description 刪除節點數據
* @Author 劉俊重
*/
@Test
public void delete() throws Exception{
//參數2:指定要刪除的版本,-1表示刪除全部版本
zkCli.delete("/zk",-1);
this.isExist();
}
/**
* @Description 更新節點數據
* @Author 劉俊重
*/
@Test
public void update() throws Exception{
Stat stat = zkCli.setData("/zk", "newtest".getBytes(), -1);
this.getData();
}
}
複製代碼
Thread.sleep(Long.MAX_VALUE);是爲了避免讓程序執行完以後立馬結束,讓它睡一會,測試監聽是否實現,同時在process回調函數中寫了收到通知的操做, zkCli.getChildren("/", true);這時若是咱們經過linux命令行操做了zookeeper操做節點就會觸發這裏的監聽事件。數據結構
如今假設有這樣一種需求:服務端節點有多個,能夠動態的上下線;須要讓任意一臺客戶端都能實時感知服務端節點的變化,進而鏈接目前可提供服務的節點。 實現思路:咱們能夠藉助於zookeeper這個第三方中間件,在每臺服務器啓動時都向zookeeper註冊服務器的節點信息(好比:/servers/server01;/servers/server02);客戶端每次調用以前都經過getChildren方法獲取最新的服務器節點信息,同時客戶端在zookeeper註冊監聽,監聽服務器節點的變化;若是某刻服務器server01下線了,zookeeper就會發出節點變化通知客戶端,回調process方法拉取最新的服務器節點信息。 服務端代碼以下:app
public class DistributeServer {
private static final String CONNECT_URL = "mini1:2181,mini2:2181,mini3:2181";
private static final int SESSION_TIME_OUT = 2000;
private static final String PARENT_NODE = "/servers";
private ZooKeeper zkCli = null;
/**
* @Description 獲取鏈接
* @Author 劉俊重
* @Date 2017/12/13
*/
public void getConnect() throws Exception{
zkCli = new ZooKeeper(CONNECT_URL, SESSION_TIME_OUT, new Watcher() {
@Override
public void process(WatchedEvent event) {
System.out.println(event.getType()+"-----------"+event.getPath());
try{
zkCli.getChildren("/", true);
}catch (Exception e){
}
}
});
}
/**
* @Description 服務器啓動時向zookeeper註冊服務信息
* @Author 劉俊重
* @Date 2017/12/13
*/
public void registerServer(String hostName) throws Exception {
String s = zkCli.create(PARENT_NODE + "/", hostName.getBytes(), ZooDefs.Ids.OPEN_ACL_UNSAFE, CreateMode.EPHEMERAL_SEQUENTIAL);
System.out.println("服務器:"+hostName+"已經註冊完畢");
}
/**
* @Description 模擬實際的業務操做
* @Author 劉俊重
* @Date 2017/12/13
*/
public void handelBusiness(String hostname) throws Exception {
System.out.println("服務器:"+hostname+"正在處理業務。。。");
Thread.sleep(Long.MAX_VALUE);
}
public static void main(String[] args) throws Exception {
DistributeServer server = new DistributeServer();
server.getConnect();
server.registerServer(args[0]);
server.handelBusiness(args[0]);
}
}
複製代碼
客戶端代碼以下:dom
/**
* @author 劉俊重
* @Description 模擬客戶端,拉取最新服務器節點列表並向zookeeper設置監聽
*/
public class DistributeClient {
private static final String CONNECT_URL = "mini1:2181,mini2:2181,mini3:2181";
private static final int SESSION_TIME_OUT = 2000;
private static final String PARENT_NODE = "/servers";
private ZooKeeper zkCli = null;
private volatile List<String> serverList = null;
/**
* @Description 獲取鏈接
* @Author 劉俊重
* @Date 2017/12/13
*/
public void getConnect() throws Exception{
zkCli = new ZooKeeper(CONNECT_URL, SESSION_TIME_OUT, new Watcher() {
@Override
public void process(WatchedEvent event) {
// 收到事件通知後的回調函數(應該是咱們本身的事件處理邏輯)
System.out.println(event.getType()+"-----------"+event.getPath());
try{
//從新更新服務器列表,而且註冊了監聽
getServerList();
}catch (Exception e){
}
}
});
}
/**
* @Description 獲取服務器子節點信息,並對父節點進行監聽
* @Author 劉俊重
*/
public void getServerList() throws Exception {
List<String> children = zkCli.getChildren(PARENT_NODE, true);
List<String> servers = new ArrayList<String>();
for(String child : children){
// child只是子節點的節點名
byte[] data = zkCli.getData(PARENT_NODE + "/" + child, false, null);
servers.add(new String(data));
}
//把servers賦值給成員變量serverList,以提供給各業務線程使用
serverList = servers;
System.out.println("節點數據:"+serverList);
}
/**
* @Description 模擬實際的業務操做
* @Author 劉俊重
* @Date 2017/12/13
*/
public void handelBusiness() throws Exception {
System.out.println("客戶端開始工做。。。");
Thread.sleep(Long.MAX_VALUE);
}
public static void main(String[] args) throws Exception {
DistributeClient client = new DistributeClient();
client.getConnect();
client.getServerList();
client.handelBusiness();
}
}
複製代碼
假設如今集羣中有50臺機器對某臺機器上的同一文件進行修改,如何才能保證這個文件不會被寫亂呢,使用java中的synchronized鎖確定是不行的,由於這個鎖是對某個程序而言的,而咱們這根本就不是在一個服務器上,怎麼會鎖的住,用zookeeper實現的分佈式鎖能夠實現。 設計思路:服務器啓動時都去zookeeper上註冊一個「短暫+序號」的znode節點(如/lock/1;/lock/2),並設置監聽父節點變化;獲取到父節點下全部子節點,並比較序號的大小;約定好比序號最小的獲取鎖,去操做某一文件,操做完成後刪除本身的節點(至關於釋放鎖),並註冊一個新的「短暫+序號」的znode節點;其它程序收到zookeeper發送的節點變化的通知以後,去比較序號的大小,看誰得到新鎖。
public class DistributedClientLock {
// 會話超時
private static final int SESSION_TIMEOUT = 2000;
// zookeeper集羣地址
private String hosts = "mini1:2181,mini2:2181,mini3:2181";
private String groupNode = "locks";
private String subNode = "sub";
private boolean haveLock = false;
private ZooKeeper zk;
// 記錄本身建立的子節點路徑
private volatile String thisPath;
/**
* 鏈接zookeeper
*/
public void connectZookeeper() throws Exception {
zk = new ZooKeeper(hosts, SESSION_TIMEOUT, new Watcher() {
@Override
public void process(WatchedEvent event) {
try {
// 判斷事件類型,此處只處理子節點變化事件
if (event.getType() == EventType.NodeChildrenChanged && event.getPath().equals("/" + groupNode)) {
//獲取子節點,並對父節點進行監聽
List<String> childrenNodes = zk.getChildren("/" + groupNode, true);
String thisNode = thisPath.substring(("/" + groupNode + "/").length());
// 去比較是否本身是最小id
Collections.sort(childrenNodes);
if (childrenNodes.indexOf(thisNode) == 0) {
//訪問共享資源處理業務,而且在處理完成以後刪除鎖
doSomething();
//從新註冊一把新的鎖
thisPath = zk.create("/" + groupNode + "/" + subNode, null, Ids.OPEN_ACL_UNSAFE,
CreateMode.EPHEMERAL_SEQUENTIAL);
}
}
} catch (Exception e) {
e.printStackTrace();
}
}
});
// 一、程序一進來就先註冊一把鎖到zk上
thisPath = zk.create("/" + groupNode + "/" + subNode, null, Ids.OPEN_ACL_UNSAFE,
CreateMode.EPHEMERAL_SEQUENTIAL);
// wait一小會,便於觀察
Thread.sleep(new Random().nextInt(1000));
// 從zk的鎖父目錄下,獲取全部子節點,而且註冊對父節點的監聽
List<String> childrenNodes = zk.getChildren("/" + groupNode, true);
//若是爭搶資源的程序就只有本身,則能夠直接去訪問共享資源
if (childrenNodes.size() == 1) {
doSomething();
thisPath = zk.create("/" + groupNode + "/" + subNode, null, Ids.OPEN_ACL_UNSAFE,
CreateMode.EPHEMERAL_SEQUENTIAL);
}
}
/**
* 處理業務邏輯,而且在最後釋放鎖
*/
private void doSomething() throws Exception {
try {
System.out.println("gain lock: " + thisPath);
Thread.sleep(2000);
} finally {
System.out.println("finished: " + thisPath);
//釋放鎖
zk.delete(this.thisPath, -1);
}
}
public static void main(String[] args) throws Exception {
DistributedClientLock dl = new DistributedClientLock();
dl.connectZookeeper();
Thread.sleep(Long.MAX_VALUE);
}
}
複製代碼
參考文檔:http://www.cnblogs.com/likehua/tag/zookeeper/
以一個簡單的例子來講明整個選舉的過程. 假設有五臺服務器組成的zookeeper集羣,它們的id從1-5,同時它們都是最新啓動的,也就是沒有歷史數據,在存放數據量這一點上,都是同樣的.假設這些服務器依序啓動,來看看會發生什麼.
那麼,初始化的時候,是按照上述的說明進行選舉的,可是當zookeeper運行了一段時間以後,有機器down掉,從新選舉時,選舉過程就相對複雜了。 須要加入數據id、leader id和邏輯時鐘。 數據id:數據新的id就大,數據每次更新都會更新id。 Leader id:就是咱們配置的myid中的值,每一個機器一個。 邏輯時鐘:這個值從0開始遞增,每次選舉對應一個值,也就是說: 若是在同一次選舉中,那麼這個值應該是一致的 ; 邏輯時鐘值越大,說明這一次選舉leader的進程更新. 選舉的標準就變成: 一、邏輯時鐘小的選舉結果被忽略,從新投票 二、統一邏輯時鐘後,數據id大的勝出 三、數據id相同的狀況下,leader id大的勝出 根據這個規則選出leader。