ZooKeeper应用场景之一 实现分布式锁

什么是ZooKeeper

ZooKeeper是一个分布式开源框架,提供了协调分布式应用的基本服务,它向外部应用暴露一组通用服务——分布式同步(Distributed Synchronization)、命名服务(Naming Service)、集群维护(Group Maintenance)等,简化分布式应用协调及其管理的难度,提供高性能的分布式服务。ZooKeeper本身可以以单机模式安装运行,不过它的长处在于通过分布式ZooKeeper集群(一个Leader,多个Follower),基于一定的策略来保证ZooKeeper集群的稳定性和可用性,从而实现分布式应用的可靠性。

1、ZooKeeper是为别的分布式程序服务的

2、ZooKeeper本身就是一个分布式程序(只要有半数以上节点存活,zk就能正常服务)

3、ZooKeeper所提供的服务涵盖:主从协调、服务器节点动态上下线、统一配置管理、分布式共享锁、统> 一名称服务等

4、虽然说可以提供各种服务,但是ZooKeeper在底层其实只提供了两个功能:

管理(存储,读取)用户程序提交的数据(类似namenode中存放的metadata); 并为用户程序提供数据节点监听服务;

ZooKeeper的特性

1、ZooKeeper:一个leader,多个follower组成的集群

2、全局数据一致:每个server保存一份相同的数据副本,client无论连接到哪个server,数据都是一致的

3、分布式读写,更新请求转发,由leader实施

4、更新请求顺序进行,来自同一个client的更新请求按其发送顺序依次执行

5、数据更新原子性,一次数据更新要么成功,要么失败

6、实时性,在一定时间范围内,client能读到最新数据

ZooKeeper的数据结构

在ZooKeeper内部数据结构是类似树状的文件层析结构,每个节点包含唯一的节点名称,以及可以对该节点做值的存储。ZooKeeper的节点有四种类型。

分别是临时节点、临时顺序节点、持久节点以及持久顺序节点。

持久节点可看作是将节点存储在硬盘上。

顺序节点会在node节点附加一个值,可以实现排序功能。

而临时节点则是当ZooKeeper连接关闭后自动删除该节点,这也就是实现分布式锁的关键。

ZooKeeper的应用场景

1、分布式锁

这个主要得益于ZooKeeper 为我们保证了数据的强一致性。锁服务可以分为两类,一个是 保持独占,另一个是 控制时序。

2、在dubbo官方文档中指定的注册中心就是ZooKeeper 。

3、数据发布与订阅(配置中心)

发布与订阅模型,即所谓的配置中心,顾名思义就是发布者将数据发布到ZK节点上,供订阅者动态获取数据,实现配置信息的集中式管理和动态更新。例如全局的配置信息,服务式服务框架的服务地址列表等就非常适合使用。

4、负载均衡

这里说的负载均衡是指软负载均衡。在分布式环境中,为了保证高可用性,通常同一个应用或同一个服务的提供方都会部署多份,达到对等服务。而消费者就须要在这些对等的服务器中选择一个来执行相关的业务逻辑,其中比较典型的是消息中间件中的生产者,消费者负载均衡。

接下来开始对分布式锁的实现(代码演示)

package neu.zookeeper;

/*

* 定义分布式锁

*/

public interface ZKlock {

public void lock();

public void unLock();

}

//重构重复代码,将重复代码交给子类执行
public abstract class ZookeeperAbstractLock implements ZKlock {
    //是在本机windows客户端安装的Zookeeper
private static final String CONNECTSTRING = "localhost:2181";
// 创建zk连接
protected ZkClient zkClient = new ZkClient(CONNECTSTRING);
protected static final String PATH = "/lock";
protected CountDownLatch countDownLatch = null;

public void lock() {
    if (getLock()) {
        System.out.println("###获取锁成功#####");
    } else {
        // 等待
        wait();
        // 重新获取锁
        lock();
    }
}
// 是否获取锁成功,成功返回true 失败返回fasle
abstract Boolean getLock();

// 等待
abstract void wait();
@Override
public void unLock() {
    if (zkClient != null) {
        zkClient.close();
        System.out.println("释放锁资源");
    }
}

public class ZookeeperDistrbuteLock extends ZookeeperAbstractLock {

@Override
Boolean getLock() {
    try {
        //试图创建一个PATH的临时节点,该节点不需要有值,为null即可
        zkClient.createEphemeral(PATH);
        //如果创建成功,则返回true,由于节点的唯一性,也保证了在连接没有关闭的情况下不会创建第二次
        return true;
    } catch (Exception e) {
        return false;
    }

}

@Override
void wait() {

    // 使用事件监听,获取到节点被删除,
    IZkDataListener iZkDataListener = new IZkDataListener() {
        // 当节点被删除
        public void handleDataDeleted(String dataPath) throws Exception {
            if (countDownLatch != null) {
                // 唤醒
                countDownLatch.countDown();
            }

        }

        // 当节点发生改变
        public void handleDataChange(String dataPath, Object data) throws Exception {

        }
    };

    // 注册节点信息
    zkClient.subscribeDataChanges(PATH, iZkDataListener);
    if (zkClient.exists(PATH)) {
        // 创建信号量
        countDownLatch = new CountDownLatch(1);
        try {
            // 等待
            countDownLatch.await();
        } catch (Exception e) {
            e.printStackTrace();
        }

    }
    // 删除事件通知
    zkClient.unsubscribeDataChanges(PATH, iZkDataListener);
}

}
--------------------------------------------------------------------------------------------------------------------Java 学习新手,如果有问题还希望各位多多指教!
下一次不定期更新Zookeeper 的其他应用场景。
感谢阅读!

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 219,110评论 6 508
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 93,443评论 3 395
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 165,474评论 0 356
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,881评论 1 295
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,902评论 6 392
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,698评论 1 305
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,418评论 3 419
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,332评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,796评论 1 316
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,968评论 3 337
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 40,110评论 1 351
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,792评论 5 346
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,455评论 3 331
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 32,003评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 33,130评论 1 272
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,348评论 3 373
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 45,047评论 2 355

推荐阅读更多精彩内容