这里,一般,都是创建临时有序子节点,怎么来创建,不难
说到协调,我首先想到的是北京很多十字路口的交通协管,他们手握着小红旗,指挥车辆和行人是不是可以通行。如果我们把车辆和行人比喻成运行在计算机中的单元(线程),那么这个协管是干什么的?很多人都会想到,这不就是锁么?对,在一个并发的环境里,我们为了避免多个运行单元对共享数据同时进行修改,造成数据损坏的情况出现,我们就必须依赖像锁这样的协调机制,让有的线程可以先操作这些资源,然后其他线程等待。对于进程内的锁来讲,我们使用的各种语言平台都已经给我们准备很多种选择。
package zhouls.bigdata.zkDemo;
import java.net.InetAddress;
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.CreateMode;
import org.apache.zookeeper.ZooDefs.Ids;
import org.junit.Test;
/**
*
* @author zhouls
*
*/
public class TestCurator {
@Test
public void test1() throws Exception {
// 1000:表示curator链接zk的时候超时时间是多少 3:表示链接zk的最大重试次数
RetryPolicy retryPolicy = new ExponentialBackoffRetry(1000, 3);
String connectString = "master:2181,slave1:2181,slave2:2181";
int sessionTimeoutMs = 5000;// 这个值只能在4000-40000ms之间 表示链接断掉之后多长时间临时节点会消失
int connectionTimeoutMs = 3000;// 获取链接的超时时间
CuratorFramework client = CuratorFrameworkFactory.newClient(
connectString, sessionTimeoutMs, connectionTimeoutMs,
retryPolicy);
client.start();// 开启客户端
InetAddress localhost = InetAddress.getLocalHost();
String ip = localhost.getHostAddress();
client.create().creatingParentsIfNeeded()// 如果父节点不存在则创建
.withMode(CreateMode.EPHEMERAL)//指定节点类型,临时节点
.withACL(Ids.OPEN_ACL_UNSAFE)// 设置节点权限信息
.forPath("/monitor/" + ip);//指定节点名称
while (true) {
;
}
}
@Test
public void test2() throws Exception {
// 1000:表示curator链接zk的时候超时时间是多少 3:表示链接zk的最大重试次数
RetryPolicy retryPolicy = new ExponentialBackoffRetry(1000, 3);
String connectString = "master:2181,slave1:2181,slave2:2181";
int sessionTimeoutMs = 5000;// 这个值只能在4000-40000ms之间 表示链接断掉之后多长时间临时节点会消失
int connectionTimeoutMs = 3000;// 获取链接的超时时间
CuratorFramework client = CuratorFrameworkFactory.newClient(
connectString, sessionTimeoutMs, connectionTimeoutMs,
retryPolicy);
client.start();// 开启客户端
InetAddress localhost = InetAddress.getLocalHost();
String ip = localhost.getHostAddress();
client.create().creatingParentsIfNeeded()// 如果父节点不存在则创建
.withMode(CreateMode.EPHEMERAL_SEQUENTIAL)// 指定节点类型,注意:临时节点必须在某一个永久节点下面
.withACL(Ids.OPEN_ACL_UNSAFE)// 设置节点权限信息
.forPath("/monitor/");// 指定节点名称
while (true) {
;
}
}
}
package zhouls.bigdata.zkDemo;
import java.io.IOException;
import java.util.ArrayList;
import java.util.Collections;
import java.util.List;
import java.util.concurrent.CountDownLatch;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.locks.Condition;
import java.util.concurrent.locks.Lock;
import org.apache.zookeeper.CreateMode;
import org.apache.zookeeper.KeeperException;
import org.apache.zookeeper.WatchedEvent;
import org.apache.zookeeper.Watcher;
import org.apache.zookeeper.ZooDefs;
import org.apache.zookeeper.ZooKeeper;
import org.apache.zookeeper.data.Stat;
/**
DistributedLock lock = null;
try {
lock = new DistributedLock("127.0.0.1:2181","test");
lock.lock();
//do something...
} catch (Exception e) {
e.printStackTrace();
}
finally {
if(lock != null)
lock.unlock();
}
//lock.closeZk();//在cleanup方法中添加
*
*/
public class DistributedLock implements Lock, Watcher{
private ZooKeeper zk;
private String root = "/locks";//根
private String lockName;//竞争资源的标志
private String waitNode;//等待前一个锁
private String myZnode;//当前锁
private CountDownLatch latch;//计数器
private int sessionTimeout = 30000;//30秒
private int waitTimeout = 30000;//等待节点失效最大时间 30秒
private List<Exception> exception = new ArrayList<Exception>();
/**
* 创建分布式锁,使用前请确认zkConnString配置的zookeeper服务可用
* @param zkConnString 127.0.0.1:2181
* @param lockName 竞争资源标志,lockName中不能包含单词lock
*/
public DistributedLock(String zkConnString, String lockName){
this.lockName = lockName;
// 创建一个与服务器的连接
try {
zk = new ZooKeeper(zkConnString, sessionTimeout, this);
Stat stat = zk.exists(root, false);
if(stat == null){
// 创建根节点
zk.create(root, new byte[0], ZooDefs.Ids.OPEN_ACL_UNSAFE,CreateMode.PERSISTENT);
}
} catch (IOException e) {
exception.add(e);
} catch (KeeperException e) {
exception.add(e);
} catch (InterruptedException e) {
exception.add(e);
}
}
/**
* zookeeper节点的监视器
*/
public void process(WatchedEvent event) {
if(this.latch != null) {
this.latch.countDown();
}
}
/**
* 获取锁
*/
public void lock() {
if(exception.size() > 0){
throw new LockException(exception.get(0));
}
try {
if(this.tryLock()){
System.out.println("Thread " + Thread.currentThread().getId() + " " +myZnode + " get lock true");
return;
}
else{
waitForLock(waitNode, waitTimeout);//等待获取锁
}
} catch (KeeperException e) {
throw new LockException(e);
} catch (InterruptedException e) {
throw new LockException(e);
}
}
/**
* 尝试获取锁
*/
public boolean tryLock() {
try {
String splitStr = "_lock_";
if(lockName.contains(splitStr))
throw new LockException("lockName can not contains \\u000B");
//创建临时有序子节点
myZnode = zk.create(root + "/" + lockName + splitStr, new byte[0], ZooDefs.Ids.OPEN_ACL_UNSAFE,CreateMode.EPHEMERAL_SEQUENTIAL);
System.err.println(myZnode + " is created ");
//取出所有子节点
List<String> subNodes = zk.getChildren(root, false);
//取出所有lockName的锁
List<String> lockObjNodes = new ArrayList<String>();
for (String node : subNodes) {
String _node = node.split(splitStr)[0];
if(_node.equals(lockName)){
lockObjNodes.add(node);
}
}
//对所有节点进行默认排序,从小到大
Collections.sort(lockObjNodes);
System.out.println(myZnode + "==" + lockObjNodes.get(0));
if(myZnode.equals(root+"/"+lockObjNodes.get(0))){
//如果是最小的节点,则表示取得锁
return true;
}
//如果不是最小的节点,找到比自己小1的节点
String subMyZnode = myZnode.substring(myZnode.lastIndexOf("/") + 1);
//获取比当前节点小一级的节点(Collections.binarySearch(lockObjNodes, subMyZnode):获取当前节点的角标)
waitNode = lockObjNodes.get(Collections.binarySearch(lockObjNodes, subMyZnode) - 1);
} catch (KeeperException e) {
throw new LockException(e);
} catch (InterruptedException e) {
throw new LockException(e);
}
return false;
}
public boolean tryLock(long time, TimeUnit unit) {
try {
if(this.tryLock()){
return true;
}
return waitForLock(waitNode,time);
} catch (Exception e) {
e.printStackTrace();
}
return false;
}
/**
* 等待获取锁
* @param lower :等待的锁
* @param waitTime 最大等待时间
* @return
* @throws InterruptedException
* @throws KeeperException
*/
private boolean waitForLock(String lower, long waitTime) throws InterruptedException, KeeperException {
Stat stat = zk.exists(root + "/" + lower,true);
//判断比自己小一个数的节点是否存在,如果不存在则无需等待锁,同时注册监听
if(stat != null){
System.out.println("Thread " + Thread.currentThread().getId() + " waiting for " + root + "/" + lower);
this.latch = new CountDownLatch(1);
this.latch.await(waitTime, TimeUnit.MILLISECONDS);
this.latch = null;
}
return true;
}
/**
* 取消锁监控
*/
public void unlock() {
try {
System.out.println(Thread.currentThread().getId()+",unlock " + myZnode);
zk.delete(myZnode,-1);
myZnode = null;
//zk.close();
} catch (InterruptedException e) {
e.printStackTrace();
} catch (KeeperException e) {
e.printStackTrace();
}
}
/**
* 关闭zk链接
*/
public void closeZk(){
try {
zk.close();
} catch (InterruptedException e) {
e.printStackTrace();
}
}
public void lockInterruptibly() throws InterruptedException {
this.lock();
}
public Condition newCondition() {
return null;
}
/**
* 自定义异常信息
* @author lenovo
*
*/
public class LockException extends RuntimeException {
private static final long serialVersionUID = 1L;
public LockException(String e){
super(e);
}
public LockException(Exception e){
super(e);
}
}
}