用zookeeper实现简单的发布订阅功能

发布方Provider的代码

package com.zhuyun.release.subscribe;

import org.apache.zookeeper.CreateMode;
import org.apache.zookeeper.WatchedEvent;
import org.apache.zookeeper.Watcher;
import org.apache.zookeeper.ZooDefs.Ids;
import org.apache.zookeeper.ZooKeeper;
import org.apache.zookeeper.data.Stat;

public class Provider {
	public static void main(String[] args) throws Exception {
		Watcher watcher = new Watcher() {
			@Override
			public void process(WatchedEvent event) {
				System.out.println("触发了" + event.getPath() + "的" + event.getType() + "事件!");				
			}
		};
		
		ZooKeeper zk = new ZooKeeper("192.168.10.203:2181", 20, watcher);
		
		Stat stat = zk.exists("/message", watcher);	
		if (stat == null) {				//假如节点不存在,则先创建节点
			zk.create("/message", "hello".getBytes(), Ids.OPEN_ACL_UNSAFE, CreateMode.PERSISTENT);
		}
		
		//向该节点发送消息
		zk.setData("/message", "hello world".getBytes(), -1);
		
		
		zk.close();
	}
}



订阅方Consumer的代码:

package com.zhuyun.release.subscribe;

import org.apache.zookeeper.KeeperException;
import org.apache.zookeeper.WatchedEvent;
import org.apache.zookeeper.Watcher;
import org.apache.zookeeper.Watcher.Event.EventType;
import org.apache.zookeeper.ZooKeeper;

public class Consumer implements Watcher{
	ZooKeeper zk;
	String hostPort;
	String znode;
	
	public Consumer(String hostPort,String znode) throws Exception{
		this.hostPort = hostPort;
		this.znode = znode;
		
		zk = new ZooKeeper(hostPort, 3000, this);
		//第一次获取节点消息,同时添加watcher
		System.out.println("消息内容:" + new String(zk.getData(znode, true, null)));
	}
	
	@Override
	public void process(WatchedEvent event) {
		if (event.getType() == EventType.NodeDataChanged) {
			try {
				//当节点消息变化时,触发该操作:获取变化后的消息,同时再添加watcher
				System.out.println("你有新的消息:" + new String(zk.getData("/message", true, null)));
				
			} catch (KeeperException e) {
				// TODO Auto-generated catch block
				e.printStackTrace();
			} catch (InterruptedException e) {
				// TODO Auto-generated catch block
				e.printStackTrace();
			}
		}
	}
	
	public static void main(String[] args) throws Exception {
		new Consumer("192.168.10.203:2181","/message");

		System.in.read();
	}


	
}

首先启动订阅方监听,再启动发布方发布消息,每次发布一条消息,订阅方都能收到该消息,结果如下:

消息内容:我是谁?
你有新的消息:hello
你有新的消息:你好
你有新的消息:hello world


  • 0
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Zookeeper实现分布式锁的一般步骤如下: 1. 在Zookeeper中创建一个临时节点,节点名称可以是锁的名称,节点数据可以是当前客户端的ID,表示该客户端获取了锁。 2. 客户端获取锁时,先检查是否已经存在该锁,如果不存在,则创建该锁;如果已经存在,则等待。 3. 当客户端释放锁时,删除该节点。 4. 其他客户端在创建节点时,如果发现该锁已经存在,则设置Watcher,等待上一个持有锁的客户端释放锁之后,重新尝试获取锁。 以下是一个简单的Java代码示例,演示了如何使用Zookeeper实现分布式锁: ``` public class DistributedLock { private static final String LOCK_BASE_PATH = "/mylock"; private static final String LOCK_NAME_PREFIX = "lock_"; private ZooKeeper zk; private String lockPath; public DistributedLock(String zkUrl) throws IOException, InterruptedException, KeeperException { this.zk = new ZooKeeper(zkUrl, 5000, null); createLockBasePath(); } private void createLockBasePath() throws KeeperException, InterruptedException { if (zk.exists(LOCK_BASE_PATH, false) == null) { zk.create(LOCK_BASE_PATH, new byte[0], ZooDefs.Ids.OPEN_ACL_UNSAFE, CreateMode.PERSISTENT); } } public void lock() throws KeeperException, InterruptedException { String path = zk.create(LOCK_BASE_PATH + "/" + LOCK_NAME_PREFIX, new byte[0], ZooDefs.Ids.OPEN_ACL_UNSAFE, CreateMode.EPHEMERAL_SEQUENTIAL); lockPath = path; while (true) { List<String> children = zk.getChildren(LOCK_BASE_PATH, false); String minChild = Collections.min(children); if (lockPath.endsWith(minChild)) { return; } else { String prevChild = children.get(children.indexOf(lockPath.substring(LOCK_BASE_PATH.length() + 1)) - 1); zk.exists(LOCK_BASE_PATH + "/" + prevChild, new LockWatcher()); } } } public void unlock() throws KeeperException, InterruptedException { zk.delete(lockPath, -1); } private class LockWatcher implements Watcher { @Override public void process(WatchedEvent event) { synchronized (this) { notifyAll(); } } } } ``` 在上述代码中,我们使用了ZooKeeper的EPHEMERAL_SEQUENTIAL节点类型来创建临时节点,并通过节点名称来实现锁。在获取锁时,会不断检查当前节点是否是最小的节点,如果不是,则等待上一个节点的Watcher通知,重新尝试获取锁。 需要注意的是,这只是一个简单的示例代码,实际应用中可能需要考虑更多的情况,比如节点的超时时间、异常处理等。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值