Zookeeper学习(八):Zookeeper的数据发布与订阅模式
http://blog.csdn.net/ZuoAnYinXiang/article/category/6104448
1.发布订阅的基本概念
1.发布订阅模式可以看成一对多的关系:多个订阅者对象同时监听一个主题对象,这个主题对象在自身状态发生变化时,会通知所有的订阅者对象,使他们能够自动的更新自己的状态。
2.发布订阅模式,可以让发布方和订阅方,独立封装,独立改变,当一个对象的改变,需要同时改变其他的对象,而且它不知道有多少个对象需要改变时,可以使用发布订阅模式
3.发布订阅模式在分布式系统的典型应用有, 配置管理和服务发现。
配置管理:是指如果集群中机器拥有某些相同的配置,并且这些配置信息需要动态的改变,我们可以使用发布订阅模式,对配置文件做统一的管理,让这些机器各 自订阅配置文件的改变,当配置文件发生改变的时候这些机器就会得到通知,把自己的配置文件更新为最新的配置
服务发现:是指对集群中的服务上下线做统一的管理,每个工作服务器都可以作为数据的发布方,向集群注册自己的基本信息,而让模型机器作为订阅方,订阅工 作服务器的基本信息,当工作服务器的基本信息发生改变时如上下线,服务器的角色和服务范围变更,监控服务器就会得到通知,并响应这些变化。
2.发布订阅的架构图
3.Manager Server的工作流程
4.Work Server的工作流程
5.发布订阅程序的结构图
6.程序代码实现
- package com.zk.subscribe;
- import java.io.BufferedReader;
- import java.io.InputStreamReader;
- import java.util.ArrayList;
- import java.util.List;
- import org.I0Itec.zkclient.ZkClient;
- import org.I0Itec.zkclient.serialize.BytesPushThroughSerializer;
- public class SubscribeZkClient {
- //需要多少个workserver
- private static final int CLIENT_QTY = 5;
- private static final String ZOOKEEPER_SERVER = "192.168.30.164:2181,192.168.30.165:2181,192.168.30.166:2181";
- //节点的路径
- private static final String CONFIG_PATH = "/config";//配置节点
- private static final String COMMAND_PATH = "/command";//命令节点
- private static final String SERVERS_PATH = "/servers";//服务器列表节点
- public static void main(String[] args) throws Exception
- {
- //用来存储所有的clients
- List<ZkClient> clients = new ArrayList<ZkClient>();
- //用来存储所有的workservers
- List<WorkServer> workServers = new ArrayList<WorkServer>();
- ManagerServer manageServer = null;
- try
- {
- ServerConfig initConfig = new ServerConfig();
- initConfig.setDbPwd("123456");
- initConfig.setDbUrl("jdbc:mysql://localhost:3306/mydb");
- initConfig.setDbUser("root");
- ZkClient clientManage = new ZkClient(ZOOKEEPER_SERVER, 5000, 5000, new BytesPushThroughSerializer());
- manageServer = new ManagerServer(SERVERS_PATH, COMMAND_PATH,CONFIG_PATH,clientManage,initConfig);
- manageServer.start();
- //根据定义的work服务个数,创建服务器后注册,然后启动
- for ( int i = 0; i < CLIENT_QTY; ++i )
- {
- ZkClient client = new ZkClient(ZOOKEEPER_SERVER, 5000, 5000, new BytesPushThroughSerializer());
- clients.add(client);
- ServerData serverData = new ServerData();
- serverData.setId(i);
- serverData.setName("WorkServer#"+i);
- serverData.setAddress("192.168.1."+i);
- WorkServer workServer = new WorkServer(CONFIG_PATH, SERVERS_PATH, serverData, client, initConfig);
- workServers.add(workServer);
- workServer.start();
- }
- System.out.println("敲回车键退出!\n");
- new BufferedReader(new InputStreamReader(System.in)).readLine();
- }finally{
- //将workserver和client给关闭
- System.out.println("Shutting down...");
- for ( WorkServer workServer : workServers )
- {
- try {
- workServer.stop();
- } catch (Exception e) {
- e.printStackTrace();
- }
- }
- for ( ZkClient client : clients )
- {
- try {
- client.close();
- } catch (Exception e) {
- e.printStackTrace();
- }
- }
- }
- }
- }
- package com.zk.subscribe;
- import java.util.List;
- import org.I0Itec.zkclient.IZkChildListener;
- import org.I0Itec.zkclient.IZkDataListener;
- import org.I0Itec.zkclient.ZkClient;
- import org.I0Itec.zkclient.exception.ZkNoNodeException;
- import org.I0Itec.zkclient.exception.ZkNodeExistsException;
- import com.alibaba.fastjson.JSON;
- public class ManagerServer {
- private String serversPath;
- private String commandPath;
- private String configPath;
- private ZkClient zkClient;
- private ServerConfig config;
- //用于监听zookeeper中servers节点的子节点列表变化
- private IZkChildListener childListener;
- //用于监听zookeeper中command节点的数据变化
- private IZkDataListener dataListener;
- //工作服务器的列表
- private List<String> workServerList;
- /**
- *
- * @param serversPath
- * @param commandPath Zookeeper中存放命令的节点路径
- * @param configPath
- * @param zkClient
- * @param config
- */
- public ManagerServer(String serversPath, String commandPath,String configPath, ZkClient zkClient, ServerConfig config) {
- this.serversPath = serversPath;
- this.commandPath = commandPath;
- this.zkClient = zkClient;
- this.config = config;
- this.configPath = configPath;
- this.childListener = new IZkChildListener() {
- //用于监听zookeeper中servers节点的子节点列表变化
- public void handleChildChange(String parentPath,List<String> currentChilds) throws Exception {
- //更新服务器列表
- workServerList = currentChilds;
- System.out.println("work server list changed, new list is ");
- execList();
- }
- };
- //用于监听zookeeper中command节点的数据变化
- this.dataListener = new IZkDataListener() {
- public void handleDataDeleted(String dataPath) throws Exception {
- }
- public void handleDataChange(String dataPath, Object data)
- throws Exception {
- String cmd = new String((byte[]) data);
- System.out.println("cmd:"+cmd);
- exeCmd(cmd);
- }
- };
- }
- public void start() {
- initRunning();
- }
- public void stop() {
- //取消订阅command节点数据变化和servers节点的列表变化
- zkClient.unsubscribeChildChanges(serversPath, childListener);
- zkClient.unsubscribeDataChanges(commandPath, dataListener);
- }
- /**
- * 初始化
- */
- private void initRunning() {
- //执行订阅command节点数据变化和servers节点的列表变化
- zkClient.subscribeDataChanges(commandPath, dataListener);
- zkClient.subscribeChildChanges(serversPath, childListener);
- }
- /*
- * 执行控制命令的函数
- * 1: list 2: create 3: modify
- */
- private void exeCmd(String cmdType) {
- if ("list".equals(cmdType)) {
- execList();
- } else if ("create".equals(cmdType)) {
- execCreate();
- } else if ("modify".equals(cmdType)) {
- execModify();
- } else {
- System.out.println("error command!" + cmdType);
- }
- }
- private void execList() {
- System.out.println(workServerList.toString());
- }
- private void execCreate() {
- if (!zkClient.exists(configPath)) {
- try {
- zkClient.createPersistent(configPath, JSON.toJSONString(config).getBytes());
- } catch (ZkNodeExistsException e) {
- //节点已经存在异常,直接写入数据
- zkClient.writeData(configPath, JSON.toJSONString(config).getBytes());
- } catch (ZkNoNodeException e) {
- //表示其中的一个节点的父节点还没有被创建
- String parentDir = configPath.substring(0,configPath.lastIndexOf('/'));
- zkClient.createPersistent(parentDir, true);
- execCreate();
- }
- }
- }
- private void execModify() {
- config.setDbUser(config.getDbUser() + "_modify");
- try {
- //回写到zookeeper中
- zkClient.writeData(configPath, JSON.toJSONString(config).getBytes());
- } catch (ZkNoNodeException e) {
- execCreate();
- }
- }
- }
- package com.zk.subscribe;
- import org.I0Itec.zkclient.IZkChildListener;
- import org.I0Itec.zkclient.IZkDataListener;
- import org.I0Itec.zkclient.ZkClient;
- import org.I0Itec.zkclient.exception.ZkNoNodeException;
- import org.I0Itec.zkclient.exception.ZkNodeExistsException;
- import com.alibaba.fastjson.JSON;
- import com.alibaba.fastjson.JSONObject;
- /**
- * 代表工作服务器
- * workServer服务器的信息
- *
- */
- public class WorkServer{
- private String serversPath;
- private String configPath;
- private ZkClient zkClient;
- private ServerConfig config;
- private ServerData serverData;
- private IZkDataListener dataListener;//数据监听器
- /**
- *
- * @param configPath 代表config节点的路径
- * @param serversPath 代表servers节点的路径
- * @param serverData 代表当前服务器的基本信息
- * @param zkClient 底层与zookeeper集群通信的组件
- * @param initconfig 当前服务器的初始配置
- */
- public WorkServer(String configPath,String serversPath,ServerData serverData,ZkClient zkClient, ServerConfig initconfig){
- this.configPath = configPath;
- this.serversPath = serversPath;
- this.serverData = serverData;
- this.zkClient = zkClient;
- this.config = initconfig;
- /**
- * dataListener 用于监听config节点的数据改变
- */
- this.dataListener = new IZkDataListener() {
- public void handleDataDeleted(String arg0) throws Exception {
- }
- /**
- * 当数据的值改变时处理的
- * Object data,这个data是将ServerConfig对象转成json字符串存入
- * 可以通过参数中的Object data 拿到当前数据节点最新的配置信息
- * 拿到这个data信息后将它反序列化成ServerConfig对象,然后更新到自己的serverconfig属性中
- */
- public void handleDataChange(String dataPath, Object data) throws Exception {
- String retJson = new String((byte[])data);
- ServerConfig serverConfigLocal = (ServerConfig)JSON.parseObject(retJson,ServerConfig.class);
- //更新配置
- updateConfig(serverConfigLocal);
- System.out.println("new work server config is:"+serverConfigLocal.toString());
- }
- };
- }
- /**
- * 服务的启动
- */
- public void start(){
- System.out.println("work server start...");
- initRunning();
- }
- /**
- * 服务的停止
- */
- public void stop(){
- System.out.println("work server stop...");
- //取消监听
- zkClient.unsubscribeDataChanges(configPath, dataListener);
- }
- /**
- * 服务器的初始化
- */
- private void initRunning(){
- registMeToZookeeper();
- //订阅config节点的改变
- zkClient.subscribeDataChanges(configPath, dataListener);
- }
- /**
- * 启动时向zookeeper注册自己
- */
- private void registMeToZookeeper(){
- //向zookeeper中注册自己的过程其实就是向servers节点下注册一个临时节点
- //构造临时节点
- String mePath = serversPath.concat("/").concat(serverData.getAddress());
- try{
- //存入是将json序列化
- zkClient.createEphemeral(mePath, JSON.toJSONString(serverData).getBytes());
- } catch (ZkNoNodeException e) {
- //父节点不存在
- zkClient.createPersistent(serversPath, true);
- registMeToZookeeper();
- }
- }
- /**
- * 当监听到zookeeper中config节点的配置信息改变时,要读取配置信息来更新自己的配置信息
- */
- private void updateConfig(ServerConfig serverConfig){
- this.config = serverConfig;
- }
- }
- package com.zk.subscribe;
- /**
- * 用于记录WorkServer(工作服务器)的配置信息
- */
- public class ServerConfig {
- private String dbUrl;
- private String dbPwd;
- private String dbUser;
- public String getDbUrl() {
- return dbUrl;
- }
- public void setDbUrl(String dbUrl) {
- this.dbUrl = dbUrl;
- }
- public String getDbPwd() {
- return dbPwd;
- }
- public void setDbPwd(String dbPwd) {
- this.dbPwd = dbPwd;
- }
- public String getDbUser() {
- return dbUser;
- }
- public void setDbUser(String dbUser) {
- this.dbUser = dbUser;
- }
- @Override
- public String toString() {
- return "ServerConfig [dbUrl=" + dbUrl + ", dbPwd=" + dbPwd + ", dbUser=" + dbUser + "]";
- }
- }
- package com.zk.subscribe;
- /**
- * 用于记录WorkServer(工作服务器)的基本信息
- */
- public class ServerData {
- private String address;
- private Integer id;
- private String name;
- public String getAddress() {
- return address;
- }
- public void setAddress(String address) {
- this.address = address;
- }
- public Integer getId() {
- return id;
- }
- public void setId(Integer id) {
- this.id = id;
- }
- public String getName() {
- return name;
- }
- public void setName(String name) {
- this.name = name;
- }
- @Override
- public String toString() {
- return "ServerData [address=" + address + ", id=" + id + ", name=" + name + "]";
- }
- }
启动SubscribeZkClient
在zookeeper客户端上输出命令
managerServer订阅了commod的变化后,输出变化
Zookeeper学习(八):Zookeeper的数据发布与订阅模式的更多相关文章
- Zookeeper应用之一:数据发布与订阅初体验
Zookeeper到底是什么?可以从Zookeeper提供的功能来理解.本篇小作文就是使用其提供的功能之一:数据发布与订阅. 需求:服务端开启多个实例提供服务,客户端使用服务.如果服务端某个服务下线或 ...
- RabbitMQ学习笔记(三) 发布与订阅
发布与订阅 在我们使用手机发送消息的时候,即可以选择给单个手机号码发送消息,也可以选择多个手机号码,群发消息. 前面学习工作队列的时候,我们使用的场景是一个消息只能被一个消费者程序实例接收并处理,但是 ...
- node.js 中 events emitter 的实现(发布、订阅模式)
const EventEmitter = require('events'); const myEmitter = new EventEmitter(); myEmitter.on('event', ...
- ZooKeeper 典型应用场景-数据发布与订阅
ZooKeeper 是一个高可用的分布式数据管理与系统协调框架.基于对 Paxos 算法的实现,使该框架保证了分布式环境中数据的强一致性,也正是基于这样的特性,使得 ZooKeeper 可以解决很多分 ...
- ZooKeeper学习之-Zookeeper简单介绍(一)
一.分布式协调技术 在给大家介绍ZooKeeper之前先来给大家介绍一种技术——分布式协调技术.那么什么是分布式协调技术?那么我来告诉大家,其实分布式协调技术主要用来解决分布式环境当中多个进程之间的同 ...
- zookeeper学习(2)----zookeeper和kafka的关系
转载: Zookeeper 在 Kafka 中的作用 leader 选举 和 follower 信息同步 如上图所示,kafaka集群的 broker,和 Consumer 都需要连接 Zookeep ...
- 知方可补不足~SQL2008中的发布与订阅模式
回到目录 作用:完成数据库与数据库的数据同步 原理:源数据库发布需要同时的表,存储过程,或者函数:目标数据库去订阅它,当源发生变化时,目标数据库自己同步,注意,由于这个过程是SQL自动完成的,所以要求 ...
- JS模式---发布、订阅模式
发布订阅模式又叫观察者模式,它定义一种一对多的依赖关系, 当一个对象的状态发生改变时,所有依赖于它的对象都将得到通知. document.body.addEventListener('click', ...
- Oracle学习(八):处理数据
1.知识点:能够对比以下的录屏进行阅读 SQL> --SQL语句 SQL> --1. DML语句(Data Manipulation Language 数据操作语言): insert up ...
随机推荐
- linux 安装mysql服务
1.检查是否已安装,grep的-i选项表示匹配时忽略大小写 rpm -qa|grep -i mysql *可见已经安装了库文件,应该先卸载,不然会出现覆盖错误.注意卸:载时使用了--nodeps选项, ...
- 执行Oracle存储过程报权限不足的解决方法
当前Oracle用户sofa拥有connect.dba.resource的角色权限,但奇怪的是却没有执行Oracle Procedure的权限.后来通过查找资料发现:如果sofa用户需要执行Proce ...
- Java中初始变量默认值
Java语言中有8种基本数据类型,基本情况汇总如下: 序号 数据类型 大小/位 封装类 默认值 可表示数据范围 1 byte(位) 8 Byte 0 -128~127 2 short(短整数) 16 ...
- SVG_style_script
1. <style type="text/css"> <![CDATA[ // ZC: 禁止所有 <text/>元素的选中 text { -webki ...
- js 实现自动调出键盘
在app中,在页面加载完成之后,给输入框添加一个focus,不能自动调出软键盘,可以用以下方式实现: //触发键盘 $("#content").on("touchstar ...
- asp.mvc 基本知识
(1)@Html.DisplayNameFor(model => model.Title)是显示列名, (2)@Html.DisplayFor(modelItem => item.Titl ...
- 解决:easygui.msgbox("Hello there!")报错:Tcl_Init error: Can't find a usable init.tcl in the following directories问题的解决
今天学习<父与子的编程之旅>,当看到运行第一个gui时(代码如下): import easygui easygui.msgbox("Hello there!") 发现报 ...
- mysql 逻辑查询语句执行顺序
一 SELECT语句关键字的定义顺序 SELECT DISTINCT <select_list> FROM <left_table> <join_type> JOI ...
- LeetCode OJ:Sum Root to Leaf Numbers(根到叶节点数字之和)
Given a binary tree containing digits from 0-9 only, each root-to-leaf path could represent a number ...
- java入门学习(1)一简介及其基础特点
分类: java基础 1.为什么java如此流行,为什么历史选择了它? 因为它拥有全新的编程思想,更接近人们的语言习惯,由于其编译器把代码编译成字节码,然后再不同的平台上运行分别用不同的虚拟机去解释字 ...