Spring Boot 整合Zookeeper组件

简介

         Zookeeper是一个Apache开源的分布式的应用,为系统架构提供协调服务。从设计模式角度来审视:该组件是一个基于观察者模式设计的框架,负责存储和管理数据,接受观察者的注册,一旦数据的状态发生变化,Zookeeper就将负责通知已经在Zookeeper上注册的观察者做出相应的反应,从而实现集群中类似Master/Slave管理模式。ZooKeeper的目标就是封装好复杂易出错的关键服务,将简单易用的接口和性能高效、功能稳定的系统提供给用户。

技术架构:

Spring Boot 2.6.3 、Zookeeper3.4.6 、 JDK1.8

导入依赖

      <dependency>
            <groupId>org.apache.curator</groupId>
            <artifactId>curator-framework</artifactId>
            <version>2.12.0</version>
        </dependency>
        <dependency>
            <groupId>org.apache.curator</groupId>
            <artifactId>curator-recipes</artifactId>
            <version>2.12.0</version>
        </dependency>
        <dependency>
            <groupId>org.apache.curator</groupId>
            <artifactId>curator-client</artifactId>
            <version>2.12.0</version>
        </dependency>
       <dependency>
            <groupId>org.projectlombok</groupId>
            <artifactId>lombok</artifactId>
            <version>1.18.4</version>
        </dependency>

application.yml配置

zoo:
  keeper:
    #开启标志
    enabled: true
    #服务器地址
    server: 192.168.116.100:2181
    #命名空间,被称为ZNode
    namespace: lx
    #权限控制,加密
    digest: smile:111111
    #会话超时时间
    sessionTimeoutMs: 3000
    #连接超时时间
    connectionTimeoutMs: 60000
    #最大重试次数
    maxRetries: 10
    #初始休眠时间
    baseSleepTimeMs: 1000

Zookeeper配置

package com.example.zk.config;

import lombok.Data;
import org.springframework.boot.context.properties.ConfigurationProperties;
import org.springframework.stereotype.Component;

/**
 * @author lanx
 * @date 2022/3/6
 */
@Data
@Component
@ConfigurationProperties(prefix = "zoo.keeper")
public class ZookeeperProperties {
    private String enabled;
    private String server;
    private String namespace;
    private String digest;
    private int sessionTimeoutMs;
    private int connectionTimeoutMs;
    private int maxRetries;
    private int baseSleepTimeMs;

}

Zookeeper 初始化

package com.example.zk.config;

import lombok.extern.slf4j.Slf4j;
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.springframework.beans.factory.annotation.Autowired;
import org.springframework.context.annotation.Configuration;

import javax.annotation.PostConstruct;

/**
 * 配置
 * @author lanx
 * @date 2022/3/6
 */
@Slf4j
@Configuration
public class ZookeeperConfig {
    @Autowired
    private ZookeeperProperties zookeeperProperties ;

    private static CuratorFramework client = null ;
    /**
     * 初始化
     */
    @PostConstruct
    public void init (){
        //重试策略,初试时间1秒,重试10次
        RetryPolicy policy = new ExponentialBackoffRetry(
                zookeeperProperties.getBaseSleepTimeMs(),
                zookeeperProperties.getMaxRetries());
        //通过工厂创建Curator
        client = CuratorFrameworkFactory.builder()
                .connectString(zookeeperProperties.getServer())
                .authorization("digest",zookeeperProperties.getDigest().getBytes())
                .connectionTimeoutMs(zookeeperProperties.getConnectionTimeoutMs())
                .sessionTimeoutMs(zookeeperProperties.getSessionTimeoutMs())
                .retryPolicy(policy).build();
        //开启连接
        client.start();
        log.info("zookeeper 初始化完成...");
    }
    public static CuratorFramework getClient (){
        return client ;
    }
    public static void closeClient (){
        if (client != null){
            client.close();
        }
    }
}

Zookeeper节点数据监听

package com.example.zk.config;

import lombok.extern.slf4j.Slf4j;
import org.apache.curator.framework.CuratorFramework;
import org.apache.curator.framework.recipes.cache.*;
import org.springframework.boot.ApplicationArguments;
import org.springframework.boot.ApplicationRunner;
import org.springframework.stereotype.Component;

import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;

/**
 * 节点数据监听
 * @author lanx
 * @date 2022/3/6
 */
@Component
@Slf4j
public class ZookListening implements ApplicationRunner {

    @Override
    public void run(ApplicationArguments args) throws Exception {
        /**
         * 在注册监听器的时候,如果传入此参数,当事件触发时,逻辑由线程池处理
         */
        ExecutorService pool = Executors.newFixedThreadPool(2);
        /**
         * 监听数据节点的变化情况
         */
        final NodeCache nodeCache = new NodeCache(ZookeeperConfig.getClient(), "/lanxi", false);
        nodeCache.start(true);
        nodeCache.getListenable().addListener(
                /**
                 * 此方法只监听创建节点和更新节点,在删除节点是不触发此操作
                 */
                new NodeCacheListener() {
                    @Override
                    public void nodeChanged() throws Exception {
                        log.info("Node data is changed, new data: {}" ,
                                new String(nodeCache.getCurrentData().getData()));
                    }
                },
                pool
        );


        /**
         * 监听子节点的变化情况
         *
         * 建立一个PathChildrenCache的缓存,第三个参数为是否接收节点数据内容,如果为 false 则不接受
         */
        final PathChildrenCache childrenCache = new PathChildrenCache(ZookeeperConfig.getClient(), "/llx", true);
        //在初始时就进行缓存监听
        childrenCache.start(PathChildrenCache.StartMode.POST_INITIALIZED_EVENT);
        childrenCache.getListenable().addListener(
                new PathChildrenCacheListener() {
                    /**
                     * 监听节点的变更。 新建、修改、删除
                     * @param client
                     * @param event
                     * @throws Exception
                     */
                    @Override
                    public void childEvent(CuratorFramework client, PathChildrenCacheEvent event)
                            throws Exception {
                        switch (event.getType()) {
                            case CHILD_ADDED:
                                System.out.println("CHILD_ADDED: " + event.getData().getPath());
                                break;
                            case CHILD_REMOVED:
                                System.out.println("CHILD_REMOVED: " + event.getData().getPath());
                                break;
                            case CHILD_UPDATED:
                                System.out.println("CHILD_UPDATED: " + event.getData().getPath());
                                break;
                            default:
                                break;
                        }
                    }
                },
                pool
        );

    }
}

curator框架,使用 DistributedAtomicInteger 作为分布式计数器

 //curator框架,使用 DistributedAtomicInteger 作为分布式计数器
        DistributedAtomicInteger atomicInteger =
                new DistributedAtomicInteger(ZookeeperConfig.getClient(), "/llx", new RetryNTimes(3, 1000));
        //复位
        atomicInteger.forceSet(0);
        AtomicValue<Integer> value = atomicInteger.add(1);
        log.info("Boolean:{}", value.succeeded());
        log.info("最新值:{}", value.postValue());
        log.info("原始值:{}", value.preValue());

curator 框架,使用 InterProcessMutex 分布式锁

 for (int i = 0; i < 10; i++) {
            new Thread(new Runnable() {
                @Override
                public void run() {
                    //分布式锁
                    final InterProcessMutex lock = new InterProcessMutex(ZookeeperConfig.getClient(), "/llx");
                    try {
                        //获取锁
                        lock.acquire();
                        log.info("线程:{},执行业务逻辑", Thread.currentThread().getName());
                        Thread.sleep(1000*10);
                    } catch (Exception e) {
                        e.printStackTrace();
                    } finally {
                        try {
                            //释放锁
                            lock.release();
                        } catch (Exception e) {
                            e.printStackTrace();
                        }
                    }
                }
            }).start();
        }

Zookeeper操作Service类

package com.example.zk.service;

import org.apache.curator.framework.recipes.locks.InterProcessReadWriteLock;
import org.apache.zookeeper.CreateMode;

import java.util.List;

public interface ZookeeperService {
    /**
     * 判断节点是否存在
     */
    boolean isExistNode (final String path) ;
    /**
     * 创建节点
     */
    void createNode (CreateMode mode, String path ) ;
    /**
     * 设置节点数据
     */
    void setNodeData (String path, String nodeData) ;
    /**
     * 创建节点
     */
    void createNodeAndData (CreateMode mode, String path , String nodeData) ;
    /**
     * 获取节点数据
     */
    String getNodeData (String path) ;
    /**
     * 获取节点下数据
     */
    List<String> getNodeChild (String path) ;
    /**
     * 是否递归删除节点
     */
    void deleteNode (String path,Boolean recursive) ;
    /**
     * 获取读写锁
     */
    InterProcessReadWriteLock getReadWriteLock (String path) ;
}
package com.example.zk.service.impl;

import com.example.zk.config.ZookeeperConfig;
import com.example.zk.service.ZookeeperService;
import lombok.extern.slf4j.Slf4j;
import org.apache.curator.framework.CuratorFramework;
import org.apache.curator.framework.recipes.locks.InterProcessReadWriteLock;
import org.apache.zookeeper.CreateMode;
import org.apache.zookeeper.data.Stat;
import org.springframework.stereotype.Service;
import org.springframework.util.StringUtils;

import java.util.ArrayList;
import java.util.List;

@Slf4j
@Service
public class ZookeeperServiceImpl implements ZookeeperService {
    @Override
    public boolean isExistNode(String path) {
        CuratorFramework client = ZookeeperConfig.getClient();
        client.sync() ;
        try {
            Stat stat = client.checkExists().forPath(path);
            return client.checkExists().forPath(path) != null;
        } catch (Exception e) {
            log.error("isExistNode error...", e);
            e.printStackTrace();
        }
        return false;
    }
    @Override
    public void createNode(CreateMode mode, String path) {
        CuratorFramework client = ZookeeperConfig.getClient() ;
        try {
            // 递归创建所需父节点
            client.create().creatingParentsIfNeeded().withMode(mode).forPath(path);
        } catch (Exception e) {
            log.error("createNode error...", e);
            e.printStackTrace();
        }
    }
    @Override
    public void setNodeData(String path, String nodeData) {
        CuratorFramework client = ZookeeperConfig.getClient() ;
        try {
            // 设置节点数据
            client.setData().forPath(path, nodeData.getBytes("UTF-8"));
        } catch (Exception e) {
            log.error("setNodeData error...", e);
            e.printStackTrace();
        }
    }
    @Override
    public void createNodeAndData(CreateMode mode, String path, String nodeData) {
        CuratorFramework client = ZookeeperConfig.getClient() ;
        try {
            // 创建节点,关联数据
            client.create().creatingParentsIfNeeded().withMode(mode)
                  .forPath(path,nodeData.getBytes("UTF-8"));
        } catch (Exception e) {
            log.error("createNode error...", e);
            e.printStackTrace();
        }
    }
    @Override
    public String getNodeData(String path) {
        CuratorFramework client = ZookeeperConfig.getClient() ;
        try {
            // 数据读取和转换
            byte[] dataByte = client.getData().forPath(path) ;
            String data = new String(dataByte,"UTF-8") ;
            if (!StringUtils.isEmpty(data)){
                return data ;
            }
        }catch (Exception e) {
            log.error("getNodeData error...", e);
            e.printStackTrace();
        }
        return null;
    }
    @Override
    public List<String> getNodeChild(String path) {
        CuratorFramework client = ZookeeperConfig.getClient() ;
        List<String> nodeChildDataList = new ArrayList<>();
        try {
            // 节点下数据集
            nodeChildDataList = client.getChildren().forPath(path);
        } catch (Exception e) {
            log.error("getNodeChild error...", e);
            e.printStackTrace();
        }
        return nodeChildDataList;
    }
    @Override
    public void deleteNode(String path, Boolean recursive) {
        CuratorFramework client = ZookeeperConfig.getClient() ;
        try {
            if(recursive) {
                // 递归删除节点
                client.delete().guaranteed().deletingChildrenIfNeeded().forPath(path);
            } else {
                // 删除单个节点
                client.delete().guaranteed().forPath(path);
            }
        } catch (Exception e) {
            log.error("deleteNode error...", e);
            e.printStackTrace();
        }
    }
    @Override
    public InterProcessReadWriteLock getReadWriteLock(String path) {
        CuratorFramework client = ZookeeperConfig.getClient() ;
        // 写锁互斥、读写互斥
        InterProcessReadWriteLock readWriteLock = new InterProcessReadWriteLock(client, path);
        return readWriteLock ;
    }
}

Zookeeper操作Controller类

package com.example.zk.web;

import com.example.zk.service.ZookeeperService;
import org.apache.zookeeper.CreateMode;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RestController;

import java.util.List;

import static org.apache.zookeeper.CreateMode.PERSISTENT;

/**
 * Zookeeper操作
 * @author lanx
 * @date 2022/3/6
 */
@RestController
public class ZookeeperApi {
    @Autowired
    private ZookeeperService zookeeperService ;

    /**
     * 获取节点下数据
     * @param path
     * @return
     */
    @GetMapping("/getNodeData")
    public String getNodeData (String path) {
        return zookeeperService.getNodeData(path) ;
    }

    /**
     * 判断节点是否存在
     * @param path
     * @return
     */
    @GetMapping("/isExistNode")
    public boolean isExistNode (final String path){
        return zookeeperService.isExistNode(path) ;
    }

    /**
     * 创建节点
     * @param path
     * @return
     */
    @GetMapping("/createNode")
    public String createNode (String path ){
        zookeeperService.createNode(CreateMode.PERSISTENT,path) ;
        return "success" ;
    }

    /**
     * 节点设置数据
     * @param path
     * @param nodeData
     * @return
     */
    @GetMapping("/setNodeData")
    public String setNodeData (String path, String nodeData) {
        zookeeperService.setNodeData(path,nodeData) ;
        return "success" ;
    }

    /**
     * 创建节点并保存数据
     * @param path
     * @param nodeData
     * @return
     */
    @GetMapping("/createNodeAndData")
    public String createNodeAndData ( String path , String nodeData){
        zookeeperService.createNodeAndData(PERSISTENT,path,nodeData) ;
        return "success" ;
    }

    /**
     *获取节点下数据
     * @param path
     * @return
     */
    @GetMapping("/getNodeChild")
    public List<String> getNodeChild (String path) {
        return zookeeperService.getNodeChild(path) ;
    }

    /**
     * 是否递归删除节点
     * @param path
     * @param recursive
     * @return
     */
    @GetMapping("/deleteNode")
    public String deleteNode (String path,Boolean recursive) {
        zookeeperService.deleteNode(path,recursive) ;
        return "success" ;
    }
}

接口测试

http://127.0.0.1:8088/createNode?path=/llx/222

  • 0
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
### 回答1: Spring Boot是一个开源的Java框架,用于快速构建独立的、可扩展的、基于生产级的Spring应用程序。它简化了Spring应用程序的配置和部署,并且提供了许多便利的特性,例如自动配置、内嵌服务器等。 Kafka是一个分布式的流处理平台,用于处理高容量的实时数据流。它使用发布-订阅的消息队列模式,可以有效地处理和存储大规模的数据流。Kafka提供了高吞吐量、持久性和容错性,适用于构建大规模的数据处理应用程序。 Zookeeper是一个分布式的协调服务,用于管理和协调大规模分布式系统的配置信息、命名服务、分布式同步和组服务等。它提供了可靠的数据存储,可以用于储存系统的元数据、配置信息等。Zookeeper还提供了分布式锁、选举、监控等功能,用于实现高可用和故障恢复能力。 Spring Boot集成了Kafka和Zookeeper,提供了方便的接口和功能,使得在Spring应用程序中使用Kafka和Zookeeper变得更加简单和高效。通过Spring Boot的自动配置,可以很容易地连接和配置Kafka和Zookeeper,并且可以方便地使用Kafka和Zookeeper的API进行消息的生产、消费和管理等操作。 总结来说,Spring Boot Kafka Zookeeper提供了一个全面的解决方案,用于构建分布式、高性能的数据处理应用程序。它使得开发者能够更加专注于业务逻辑的实现,而不用过多关注底层的配置和部署。通过使用Spring Boot Kafka Zookeeper,可以简化开发过程、提高开发效率,并且可以构建出高可伸缩、高可靠性的应用程序。 ### 回答2: Spring Boot是一个用于构建Java应用程序的开源框架,它提供了快速、方便和可重复使用的方式来开发基于Java的企业级应用程序。Kafka是一个分布式流处理平台,它可以用来构建高度可扩展的、实时的数据流应用程序。而Zookeeper是一个开源的分布式协调服务框架,它提供了一个分布式的、高可用性的系统,用于协调和管理大规模分布式应用程序的配置、命名服务和分布式锁等。 使用Spring Boot与Kafka和Zookeeper结合使用,可以构建高效且可伸缩的分布式应用程序。首先,Spring Boot可以通过提供简化的配置和自动化的依赖管理,简化Kafka和Zookeeper的集成过程。同时,Spring Boot还提供了许多与Kafka和Zookeeper交互的高级抽象,如KafkaTemplate用于发送和接收消息,以及CuratorFramework用于与Zookeeper进行交互。 使用Spring Boot与Kafka结合,可以实现实时的数据流处理。Kafka提供了可靠的、高吞吐量的消息传递,而Spring Boot通过提供简单的注解和配置,使得在应用程序中发送和接收消息变得非常容易。这种组合适用于实时分析、流媒体处理和事件驱动的应用程序等场景。 在与Zookeeper结合使用时,Spring Boot可以通过提供CuratorFramework的高级抽象,使得管理分布式应用程序的配置、命名服务和分布式锁等变得简单。Zookeeper可以作为底层的分布式系统,确保应用程序的高可用性和可靠性。 总而言之,Spring Boot与Kafka和Zookeeper的结合可以极大地简化分布式应用程序的开发和管理过程,减少了与底层技术的交互复杂性。它们提供了一种方便且可靠的方式来构建高性能、可伸缩和高可用性的分布式应用程序。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

终遇你..

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值