zookeeper 集群搭建教程之应用案例

       Apache ZooKeeper是Apache软件基金会的一个软件项目,它为大型分布式计算提供开源的分布式配置服务、同步服务和命名注册。ZooKeeper曾经是Hadoop的一个子项目,但现在是一个独立的顶级项目。ZK没有直接采用Paxos算法的实现,而是使用了一种称为Zab(Zookeeper Atomic Broadcast 原子消息广播协议)的一致性协议
    


         本文目录:
             1、zookeeper的简介与应用场景
             2、zookeeper的设计目标
             3、ZAB协议
             4~6、zookeeper的搭建(集群、伪集群、单机)
             7、zookeeper客户端的脚本使用
             8测试zookeeper的功能(选举、共享文件)
             9、客户端操作(JavaAPI、ZKClient、Curator)
             
             


1、Zookeeper应用场景

      是一个典型的分布式数据一致性的解决方案,分布式应用程序可以基于它实现数据发布/订阅、负载均衡、分布式协调/通知、集群管理、Master选举、分布式锁和分布式队列等。



2、zookeeper的设计目标

     致于是提供一个高性能、高可用,并具有严格的顺序访问控制能力的分布式协调服务。


    目标一、简单的数据模型:使得分布式程序能够通过一个共享的、树型结构的名字空间来进行互相协调。


    目标二、可构建集群:组成zk集群的每台机器都会在内存中维护当前的服务状态,并且每台机器之间都能相互保持通。


集群结构图



    目标三、顺序访问:客户的每一个更新请求、zk都会分配一个全局唯一的递增编号,反映所以实务操作的先后顺序。


    目标四、高性能:由于zk将全量数据存储在内中,并且直接服务于客户端的所以非事务请求,因此它尤其适用于读操作作为主的应用场景。




3、ZAB协议

     ZAB协议的两种基本模式

     消息广播:Leader服务器会为每个事务请求生成对应proposal来进行广播,并且在广播事务proposal之前、leader服务器会去为这个事务proposal分配一个全局单调递增的唯一ID,事务proposal按照先后顺序进行排序与处理。当每一个follower服务器在接收到这个事务proposal之后,就将其以日志的形式写入到本地磁盘中。在成功写入后反馈给leader服务器一个AC响应。

     崩溃恢复:当leader服务器崩溃或者其他原因不能与follower联系、ZAB协议 就需要一个高效并且可靠的leader选举算法,从而选出新的leader。






4、搭建zookeeper

     这里将介绍使用zookeeper搭建集群、伪分布式搭建、单机及其客户端的使用。

     系统环境:zookeeper支持很多操作系统,我用的Linux(CentOS - 7)。

     语言环境:zookeeper是Java语言编写的,所以需要Java环境的支持。


图-1

4.1、搭建集群(3台CentOS-7)

  有的朋友就会说,我的虚拟机开不了那么多,不用怕。下面有伪分布式搭建教程。



4.2、zookeeper搭建步骤:

1、获取zookeeper包
     到 http://apache.fayea.com/zookeeper/下载 zookeeper-3.4.6:
     $ wget http://apache.fayea.com/zookeeper/zookeeper-3.4.6/zookeeper-3.4.6.tar.gz

     本地上传(我的选择)

2、解压zookeeper 安装包
    $ tar -zxvf zookeeper-3.4.6.tar.gz

3、移动并命名(喜欢放哪都好)
    $mv  zookeeper-3.4.6  zookeeper
    $mv  zookeeper   /usr/local/chenzhengyou/

3、配置
     zookeeper 目录下创建以下目录:
     $ cd  /usr/local/chenzhengyou/zookeeper
     $ mkdir data
     $ mkdir logs

4、 将 zookeeper/conf 目录下的 zoo_sample.cfg 文件拷贝一份,命名为为zoo.cfg
     $ cp  zoo_sample.cfg zoo.cfg

5、修改 zoo.cfg 配置文件(如图-2)
     $ vim  zoo.cfg

图-2
6、 在/usr/local/chenzhengyou/zookeeper/data 下创建 myid 文件编辑 myid 文件,并在对应的 IP 的机器上输入对应的编号。如在 zookeeper 上,myid文件内容就是 1(目前该ip后缀100作为1号机器)。
      $ vim  myid

7、
为了方便关闭防火墙(3台都关了
     $ sudo systemctl stop firewalld.service


8、使用scp远程传输(改ip再把传给103机器
     $ scp -r zookeeper/ root@192.168.10.102:/usr/local/chenzhengyou/

9、只需修改/usr/local/chenzhengyou/zookeeper/data目录下的 myid 文件(看步骤6
     102机器myid文件2
     103机器myid文件3

10、启动并测试 zookeeper

     (1) 到/usr/local/chenzhengyou/zookeeper/bin目录中执行(每台机器都启动

          $ ./zkServer.sh start


     (2) 输入 jps 命令查看进程:
          $ jps
              1456 QuorumPeerMain
              1475 Jps

         其中, QuorumPeerMain 是 zookeeper 进程,启动正常
     

     (3) 查看状态
        $ ./zkServer.sh status(会发现有一个主节点、两个从节点





5、zookeeper伪分布式搭建

   只需修改zoo.cfg文件

server.1=192.168.10.100:2888:3888
server.2=192.168.10.100:2889:3889
server.3=192.168.10.100:2890:3890


6、zookeeper单机搭建

  只需修改(只有IP配置就好了)
server.1=192.168.10.100:2888:3888



7、zookeeper客户端脚本的使用

   启动客户端
   $ ./zkCli.sh


7.1、读取(这是我以前创建的节点)

   格式:$ ls  path  [watch] 、path表示数据节点节点的路径。
   执行如下命令(效果如图)
   $ ls /




7.2、创建与读取

  格式:$create  [ -s ] [ -e ] path  data  acl
  其中,  [ -s ] 或[ -e ]分别表示节点特性:顺序节点或者临时节点,默认不添加[ -s ] 或[ -e ]则表示持久节点。



7.3、更新

  格式:$set  path  data  [ version ]
  其中,data是要更新的内容,节点的数据是有版本的概念的,便于操作哪一个版本的数据。仔细观察会发现版本的变化。




7.4、删除

  格式:$ delete  path  [ version ]



8、测试zookeeper的功能

     这里在集群测试zookeeper数据能不能共享机制,在一个节点上创建一个文件。在另一个节点有没有该文件?答案是可以的。如下、三台机器上的节点完全一样。



   
   如果我把 原来的leader down掉、它们之间会不会自动选一个来当 leader呢?答案是可以的。
   


9.1、JavaAPI简单操作zookeeper

package com.czy.zookeeper.base;

import java.util.concurrent.CountDownLatch;

import org.apache.zookeeper.CreateMode;
import org.apache.zookeeper.WatchedEvent;
import org.apache.zookeeper.Watcher;
import org.apache.zookeeper.Watcher.Event.EventType;
import org.apache.zookeeper.ZooKeeper;
import org.apache.zookeeper.Watcher.Event.KeeperState;
import org.apache.zookeeper.ZooDefs.Ids;


/**
 * @auther 陈郑游
 * @create
 * @功能   Zookeeper
 * @问题
 * @说明
 * @URL地址
 * @进度描述
 */
public class ZookeeperBase {

	/** zookeeper地址 */
	static final String CONNECT_ADDR = "192.168.100.11:2181,192.168.100.12:2181,192.168.100.13:2181";
	/** session超时时间秒 */
	static final int SESSION_OUTTIME = 5000;
	/** 信号量,阻塞程序执行,用于等待zookeeper连接成功,发送成功信号 */
	static final CountDownLatch connectedSemaphore = new CountDownLatch(1);



	public static void main(String[] args) throws Exception{

		ZooKeeper zk = new ZooKeeper(CONNECT_ADDR, SESSION_OUTTIME, new Watcher() {
			@Override
			public void process(WatchedEvent watchedEvent) {
				//获取事件的状态
				KeeperState keeperState = watchedEvent.getState();
				EventType eventType = watchedEvent.getType();
				//如果是建立连接
				if(KeeperState.SyncConnected == keeperState){
					if(EventType.None == eventType){
						//如果建立连接成功,则发送信号量,让后续阻塞程序向下执行
						connectedSemaphore.countDown();
						System.out.println("ZK已经建立连接!");
					}
				}
			}
		});
		//进行阻塞
		connectedSemaphore.await();
		
		System.out.println("...................");

		//创建父节点
//		zk.create("/Root", "Rootdata".getBytes(), Ids.OPEN_ACL_UNSAFE, CreateMode.PERSISTENT);

		//创建子节点
//		zk.create("/Root/children", "children data".getBytes(), Ids.OPEN_ACL_UNSAFE, CreateMode.PERSISTENT);
		
		//获取节点洗信息
		byte[] data = zk.getData("/", false, null);
		System.out.println(new String(data));
		System.out.println(zk.getChildren("/", false));
		
		//修改节点的值
//		zk.setData("/testRoot", "modify data root".getBytes(), -1);
//		byte[] data = zk.getData("/Root", false, null);
//		System.out.println(new String(data));		
		
		//判断节点是否存在
//		System.out.println(zk.exists("/Root/children", false));

		//删除节点
//		zk.delete("/Root/children", -1);
//		System.out.println(zk.exists("/Root/children", false));
		
		zk.close();

	}
}



9.2、ZKClient客户端简单操作zookeeper

package com.czy.zkCilent.base;

import org.I0Itec.zkclient.ZkClient;
import org.I0Itec.zkclient.ZkConnection;

import java.util.List;


/**
 * @auther 陈郑游
 * @create 2017/4/4 0004
 * @功能
 * @问题
 * @说明
 * @URL地址
 * @进度描述
 */
public class ZkClientBase {

	/** zookeeper地址 */
	static final String CONNECT_ADDR = "192.168.100.11:2181,192.168.100.12:2181,192.168.100.13:2181";
	/** session超时时间秒 */
	static final int SESSION_OUTTIME = 5000;
	
	
	public static void main(String[] args) throws Exception {
		ZkClient zkc = new ZkClient(new ZkConnection(CONNECT_ADDR), 5000);
		//1. create and delete方法 
		zkc.createEphemeral("/temp");
		zkc.createPersistent("/super/c1", true);
		Thread.sleep(10000);
		zkc.delete("/temp");
		zkc.deleteRecursive("/super");
		
		//2. 设置path和data 并且读取子节点和每个节点的内容
		zkc.createPersistent("/super", "1234");
		zkc.createPersistent("/super/c1", "c1内容");
		zkc.createPersistent("/super/c2", "c2内容");
		List<String> list = zkc.getChildren("/super");
		for(String p : list){
			System.out.println(p);
			String rp = "/super/" + p;
			String data = zkc.readData(rp);
			System.out.println("节点为:" + rp + ",内容为: " + data);
		}
		
		//3. 更新和判断节点是否存在
//		zkc.writeData("/super/c1", "新内容");
//		System.out.println(zkc.readData("/super/c1"));
//		System.out.println(zkc.exists("/super/c1"));
		
		//4.递归删除/super内容
//		zkc.deleteRecursive("/super");		
	}
}


9.3、Curator客户端简单操作zookeeper(实际是多用Curator操作)

package com.czy.curator.base;

import org.apache.curator.RetryPolicy;
import org.apache.curator.framework.CuratorFramework;
import org.apache.curator.framework.CuratorFrameworkFactory;
import org.apache.curator.retry.ExponentialBackoffRetry;
import org.apache.zookeeper.ZooKeeper;
import org.apache.zookeeper.data.Stat;

import java.util.List;


public class CuratorBase {

	/** zookeeper地址 */
	static final String CONNECT_ADDR = "192.168.100.11:2181,192.168.100.12:2181,192.168.100.13:2181";
	/** session超时时间秒 */
	static final int SESSION_OUTTIME = 5000;
	
	public static void main(String[] args) throws Exception {
		
		//1 重试策略:初试时间为1s 重试10次
		RetryPolicy retryPolicy = new ExponentialBackoffRetry(1000, 10);
		//2 通过工厂创建连接
		CuratorFramework cf = CuratorFrameworkFactory.builder()
					.connectString(CONNECT_ADDR)
					.sessionTimeoutMs(SESSION_OUTTIME)
					.retryPolicy(retryPolicy)
//					.namespace("super")
					.build();
		//3 开启连接
		cf.start();
		
		System.out.println(ZooKeeper.States.CONNECTED);
		System.out.println(cf.getState());
		
		// 新加、删除
		/**
		//4 建立节点 指定节点类型(不加withMode默认为持久类型节点)、路径、数据内容
		cf.create().creatingParentsIfNeeded().withMode(CreateMode.PERSISTENT).forPath("/super/c1","c1内容".getBytes());
		//5 删除节点
		cf.delete().guaranteed().deletingChildrenIfNeeded().forPath("/super");
		*/
		
		// 读取、修改
		/**
		//创建节点
//		cf.create().creatingParentsIfNeeded().withMode(CreateMode.PERSISTENT).forPath("/super/c1","c1内容".getBytes());
//		cf.create().creatingParentsIfNeeded().withMode(CreateMode.PERSISTENT).forPath("/super/c2","c2内容".getBytes());
		//读取节点
		String ret1 = new String(cf.getData().forPath("/super/c2"));
		System.out.println(ret1);

		//修改节点
//		cf.setData().forPath("/super/c2", "修改c2内容".getBytes());
//		String ret2 = new String(cf.getData().forPath("/super/c2"));
//		System.out.println(ret2);	
		*/
		
		// 绑定回调函数
		/**
		ExecutorService pool = Executors.newCachedThreadPool();
		cf.create().creatingParentsIfNeeded().withMode(CreateMode.PERSISTENT)
		.inBackground(new BackgroundCallback() {
			@Override
			public void processResult(CuratorFramework cf, CuratorEvent ce) throws Exception {
				System.out.println("code:" + ce.getResultCode());
				System.out.println("type:" + ce.getType());
				System.out.println("线程为:" + Thread.currentThread().getName());
			}
		}, pool)
		.forPath("/super/c3","c3内容".getBytes());
		Thread.sleep(Integer.MAX_VALUE);
		*/
		
		
		// 读取子节点getChildren方法 和 判断节点是否存在checkExists方法
		/***/
		List<String> list = cf.getChildren().forPath("/super");
		for(String p : list){
			System.out.println(p);
		}
		
		Stat stat = cf.checkExists().forPath("/super/c3");
		System.out.println(stat);
		
		Thread.sleep(2000);
		cf.delete().guaranteed().deletingChildrenIfNeeded().forPath("/super");

		
		
		//cf.delete().guaranteed().deletingChildrenIfNeeded().forPath("/super");
		
	}
}



















  • 2
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值