全局式分布式锁要求任何时刻没有两个客户端会获得同一个锁对象,这可以通过使用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. Springmvc数据校验

    步骤一:导入四个jar包 <?xml version="1.0" encoding="UTF-8"?> <beans xmlns=" ...

  2. iPhone Anywehre虚拟定位提示“后台服务未启动,请重新安装应用后使用”的解决方法

    问题描述: iPhone越狱了,之后在Cydia中安装Anywhere虚拟定位,但是打开app提示:后台服务未启动,请重新安装应用后使用. 程序无法正常使用... 解决方法: 打开Cydia-已安装, ...

  3. 【SQLServer】记一次数据迁移-标识重复的简单处理

    汇总篇:http://www.cnblogs.com/dunitian/p/4822808.html#tsql 今天在数据迁移的时候因为手贱遇到一个坑爹问题,发来大家乐乐,也传授新手点经验 迁移惯用就 ...

  4. Vue + Webpack + Vue-loader 系列教程(1)功能介绍篇

    原文地址:https://lvyongbo.gitbooks.io/vue-loader/content/ Vue-loader 是什么? vue-loader 是一个加载器,能把如下格式的 Vue ...

  5. [OpenGL超级宝典]专栏前言

    我小时候的梦想呢,是做宇航员或者科学家或者是做一款属于自己的游戏,后来前面两个梦想都没有实现,于是我就来实现我的第三个梦想了,,,我呢,也算是零基础,因为我的专业是物联网工程,这个专业覆盖面之广,简直 ...

  6. C#创建dll类库

    类库让我们的代码可复用,我们只需要在类库中声明变量一次,就能在接下来的过程中无数次地使用,而无需在每次使用前都要声明它.这样一来,就节省了我们的内存空间.而想要在类库添加什么类,还需取决于类库要实现哪 ...

  7. .NET面试题集锦①(Part一)

    一.前言部分 文中的问题及答案多收集整理自网络,不保证100%准确,还望斟酌采纳. 1.面向对象的思想主要包括什么? 答:任何事物都可以理解为对象,其主要特征: 继承.封装.多态.特点:代码好维护,安 ...

  8. CSS三个定位——常规、浮动、绝对定位

    .dage { width: 868px; background: #5B8C75; border: 10px solid #A08C5A; margin-top: -125px; margin-le ...

  9. BPM配置故事之案例13-触发消息通知

    老李:小明! 小明:--见你就没好事,又要我干嘛? 老李:额,小事小事,最近很多部门都觉得Boss的审批速度太慢了,能不能以后给审批人一个消息提醒? 小明:--有一种不太好的预感 老李:怎么,很困难么 ...

  10. iOS中支付宝集成

    iOS中支付宝集成 如今各种的App中都使用了三方支付的功能,现在将我在使用支付宝支付集成过程的心得分享一下,希望对大家都能有所帮助 要集成一个支付宝支付过程的环境,大致需要: 1>公司:先与支 ...