全局式分布式锁要求任何时刻没有两个客户端会获得同一个锁对象,这可以通过使用ZooKeeper实现。像优先级队列一样,首先需要定义一个锁节点。

在ZooKepeer的发布中src/recipes/lock(https://github.com/apache/zookeeper/tree/master/src/recipes/lock)的目录有ZooKeeper的Lock实现。

要获得锁的客户端进行如下操作:

1.调用Create方法并使用"_locknode_/guid-lock-"作为路径,并设置sequence和ephemeral标志位。guid是防止create的结果缺失,详见下面说明。

2.在锁节点上调用getChildren方法,不用设置watch(这是为了防止羊群效应)。

3.如果第一步所创建的路径拥有最小的序号,该客户端获得锁并退出协议。

4.客户端调用exists并设置watch标志在比自身小的节点上。

5.如果exists返回false,返回第二步。不然等到上一步设置的watch事件触发再回到第二步。

解锁的步骤非常简单:客户端希望释放锁只需要简单地删除第一步所创建的锁。

这里有几点值得注意:

  • 由于每个节点只会被一个客户端watch,所以删除一个节点只会唤醒一个客户端。这样可以避免羊群效应。
  • 没有轮询或超时的问题
  • 使用该方式实现锁可以方便地看见锁的竞争、中断锁占用、调试锁等

可恢复错误和GUID

  • 如果调用create时发生了一个可恢复的错误,客户端可以调用getChildren并通过guid检查节点的名字,来得知自己创建的节点。这个解决了引言中提到的问题,即create成功返回但是服务器在获得节点名字之前宕机的情况。

共享锁

可以通过修改部分协议来实现共享锁。

获得读锁 获得写锁
1.调用create创建"guid-/read-"节点。这是将会用到的读锁。需要确保使用sequence和ephemeral标志位。

2.在锁节点上调用getChildren,注意不要设置watch标志,避免羊群效应。

3.如果没有比第一步创建的节点小的"write-"开头的节点,则客户端获得锁并退出协议。

4.否则在次小的write节点上调用exists方法并设置watch标志。

5.如果exists方法返回false则返回第二步。

6.不然等待上面watch事件触发再返回第二步
1.调用create创建"guid-/write-"节点。这是后面需要的写锁。需要确保使用sequence和ephemeral标志位。

2.在锁节点上调用getChildren,注意不要设置watch标志,避免羊群效应。

3.如果没有比第一步创建的更小的节点,客户端获得锁并退出协议。

4.在次小的节点上调用exists方法并设置watch标志。

5.如果exists返回false回到第二步,不然等到watch事件触发再返回第二步。

###注意:
>* 该recipe可能产生羊群效应,即,当有一群客户端等待读锁时,客户端会在写锁释放时一起被唤醒。事实上,这个过程应该是正确的:所有等待的读客户端获得了锁。而羊群效应是指有一群节点被唤醒,而事实上只能允许少数可以运行。
>* 详见[上面](#可恢复错误和guid)提到的关于的guid使用。
##可恢复的共享锁
对共享锁协议做一些小幅改动就可以实现锁的可恢复:
在上面读写锁的第一步,在调用create之后,立即调用getData并使用watch标识,如果客户端收到这个watch事件,再次调用getData并使用watch标识并查看内容是否包含"unlock"字符串,unlock表示客户端必须释放当前锁。根据共享锁协议,你可以向获得锁客户端调用setData写入"unlock"字符串,申请其释放锁。
注意这个协议需要锁持有者同意释放锁。这很重要,尤其是如果锁持有者需要在释放之前做某些操作时。当然你可以在你的协议中规定,在锁持有者一定时间后如仍未删除锁节点,可以由他人强制删除。
##Curator实现
###共享锁
```
public InterProcessSemaphoreMutex(CuratorFramework client,String path)
/**
client 客户端实例
path 锁节点地址
**/
public void acquire() //获得锁
public void release() //释放锁
```
###可重入共享锁
```
public InterProcessMutex(CuratorFramework client,String path)
/**
client 客户端实例
path 锁节点地址
**/
public void acquire() //获得锁
public void release() //释放锁
public void makeRevocable(RevocationListener listener) //撤销锁
public static void attemptRevoke(CuratorFramework client,String path) //请求对应路径锁撤销
```
###可重入读写锁
```
public InterProcessReadWriteLock(CuratorFramework client,String basePath)
/**
client 客户端实例
basePath 锁基础路径
**/
public InterProcessLock readLock() //读锁
public InterProcessLock writeLock() //写锁
```
[返回引言](http://www.cnblogs.com/resentment/p/6129339.html)

[译]ZOOKEEPER RECIPES-Locks的更多相关文章

  1. ZooKeeper Recipes and Solutions 翻译

    ZooKeeper 秘诀 与解决方案 A Guide to Creating Higher-level Constructs with ZooKeeper Out of the Box Applica ...

  2. ZooKeeper Recipes and Solutions

    原文地址:http://zookeeper.apache.org/doc/current/recipes.html 参考:https://zookeeper.apache.org/doc/trunk/ ...

  3. [译]ZooKeeper recipes-引言

    ZooKeeper高级应用 本系列将指导使用ZooKeeper来实现高级功能,所有功能都在客户端完成,不需要ZooKeeper的特殊支持.希望可以得到社区的支持将这些加入到一个标准的客户端类库中(Cu ...

  4. [译]ZOOKEEPER RECIPES-Queues

    队列 分布式队列是一种常见的数据结构.为了在ZooKepeer中实现分布式队列,第一步是要使用一个znode代表队列本身.分布式客户端通过create()方法将内容放入一个名叫"queue- ...

  5. [译]ZOOKEEPER RECIPES-Leader Election

    选主 使用ZooKeeper选主的一个简单方法是,在创建znode时使用Sequence和Ephemeral标志.主要思想是,使用一个znode,比如"/election",每个客 ...

  6. [译]ZOOKEEPER RECIPES-TWO PHASED COMMIT

    两段式提交 两段式提交协议可以让所有分布式系统中的客户端达成协议同时提交或回滚事务. 在ZooKeeper中你可以通过协调者(coordinator)创建一个事务节点来实现两段式提交.例如" ...

  7. [译]ZOOKEEPER RECIPES-Barriers

    Barrier 在分布式系统中常使用Barrier来阻塞进程,当满足一定条件后再恢复进行后续操作.Barrier在Zookeeper中可以通过设计一个Barrier节点来实现.Barrier 节点存在 ...

  8. <译>Zookeeper官方文档

    apache原文地址:http://zookeeper.apache.org/doc/trunk/zookeeperOver.html ZooKeeper ZooKeeper: A Distribut ...

  9. [译]Zookeeper的优点与局限性

    1.Zookeeper的优点与局限性 在学习了Zookeeper(后文都简称zk)的介绍和功能后,您已经很好地理解了zk. 现在,在这个zk教程中,我们将讨论zk的优点和局限性. zk有几个功能对用户 ...

随机推荐

  1. Solr_全文检索引擎系统

    Solr介绍: Solr 是Apache下的一个顶级开源项目,采用Java开发,它是基于Lucene的全文搜索服务.Solr可以独立运行在Jetty.Tomcat等这些Servlet容器中. Solr ...

  2. [翻译]开发文档:android Bitmap的高效使用

    内容概述 本文内容来自开发文档"Traning > Displaying Bitmaps Efficiently",包括大尺寸Bitmap的高效加载,图片的异步加载和数据缓存 ...

  3. javascript之活灵活现的Array

    前言 就如同标题一样,这篇文章将会灵活的运行Array对象的一些方法来实现看上去较复杂的应用. 大家都知道Array实例有这四个方法:push.pop.shift.unshift.大家也都知道 pus ...

  4. RIFF和WAVE音频文件格式

    RIFF file format RIFF全称为资源互换文件格式(Resources Interchange File Format),是Windows下大部分多媒体文件遵循的一种文件结构.RIFF文 ...

  5. WebApi基于Token和签名的验证

    最近一段时间在学习WebApi,涉及到验证部分的一些知识觉得自己并不是太懂,所以来博客园看了几篇博文,发现一篇讲的特别好的,读了几遍茅塞顿开(都闪开,我要装逼了),刚开始读有些地方不理解,所以想了很久 ...

  6. 微信小程序开发—快速掌握组件及API的方法

    微信小程序框架为开发者提供了一系列的组件和API接口. 组件主要完成小程序的视图部分,例如文字.图片显示.API主要完成逻辑功能,例如网络请求.数据存储.音视频播放控制,以及微信开放的微信登录.微信支 ...

  7. javaScript生成二维码(支持中文,生成logo)

    资料搜索 选择star最多的两个 第一个就是用的比较多的jquery.qrcode.js(但不支持中文,不能带logo)啦,第二个支持ie6+,支持中文,根据第二个源代码,使得,jquery.qrco ...

  8. 热修复-Nuwa学习篇

    nuwa热修复是基于qq空间团队的思路,最近的热度话题了,很多种方案,自己先研究几种方案,基本上都各有优势,学习肯定得先挑个软柿子捏了,自己对比了一下,发现nuwa代码量少点,所以就决定了,先研究nu ...

  9. 什么是英特尔® Edison 模块?

    英特尔® Edison 模块 是一种 SD 卡大小的微型计算芯片,专为构建物联网 (IoT) 和可穿戴计算产品而设计. Edison 模块内含一个高速的双核处理单元.集成 Wi-Fi*.蓝牙* 低能耗 ...

  10. NYOJ 975

    这道题一开始本着很朴素的想法就是先输入两头的数据,然后对每组的数据范围下测试中间的数据即可,但是是超时的.原因也很明显,比如计算1~1000的数据之后,假如下一组数据是1~1001,本来只需要多测试下 ...