Go 版本 Etcd 客户端操作 Etcd(附完整 Demo)

Etcd 相关参考资料

Etcd 的介绍与使用:Etcd 介绍与使用(入门篇)-CSDN博客

Etcd Raft 协议:Etcd Raft 协议(进阶篇)-CSDN博客

本文诣在使用 Go 客户端操作 Etcd,并实现元数据的写入(单条写、批量写)、读取(单挑读、前缀读)、监听(watch)、更新(update)!

注:数据是以 Protobuf 格式存储到 Etcd 的!

Protobuf 的介绍与使用:Protobuf 的介绍与使用(入门级)-CSDN博客

完整代码

./src/message/metadata.proto

syntax = "proto3";
package message;option go_package = "/data/etcd_test/src/message";message Metadata {string Name = 1;int64 ShardCount = 2;
}

 ./src/etcd/main.go

package mainimport ("errors""strings"//"encoding/json""etcd_test/src/message""fmt""github.com/golang/protobuf/proto""go.etcd.io/etcd/client/v3""golang.org/x/net/context""time"
)// 连接etcd集群
func connectToEtcd(args ...string) (*clientv3.Client, error) {if len(args) == 0 {return nil, errors.New("ip or port is nil")}var endpoints []stringfor _, arg := range args {endpoints = append(endpoints, arg)}// 连接etcd集群cli, err := clientv3.New(clientv3.Config{Endpoints:   endpoints,DialTimeout: 5 * time.Second,})if err != nil {return nil, errors.New("connect etcd fail")}return cli, nil
}// 初始化元数据
func initMetadata() (map[string]*message.Metadata, error) {// 初始化一个切片用于存储元数据metadata := make(map[string]*message.Metadata)// 初始化meta_tables实例data1 := &message.Metadata{Name: "dist_by_mm", ShardCount: 12}keyName1 := "meta_table"metadata[keyName1] = data1data2 := &message.Metadata{Name: "dist_by_mm", ShardCount: 14}keyName2 := "meta_table1"metadata[keyName2] = data2if len(metadata) == 0 {return nil, errors.New("metadata init error")}return metadata, nil
}// 将元数据序列化为protobuf并存储到etcd(单条循环写入)
func putMetadataToEtcdSingle(cli *clientv3.Client, metadata map[string]*message.Metadata) error {if len(metadata) == 0 {return errors.New("metadata list is nil")}// 遍历元数据列表,逐个写入etcdfor k, v := range metadata {// 将元数据序列化为protobufprotoData, err := proto.Marshal(v)if err != nil {return errors.New("metadata to protobuf fail")}// 存储protobuf数据到etcdctx, cancel := context.WithTimeout(context.Background(), 5*time.Second)defer cancel()_, err = cli.Put(ctx, k, string(protoData))if err != nil {return errors.New("metadata write fail")}}return nil
}// 将元数据序列化为protobuf并存储到etcd(批量写入)
func putMetadataToEtcdBatch(cli *clientv3.Client, metadata map[string]*message.Metadata) error {// 创建事务txn := cli.Txn(context.Background())meta := make(map[string]string)for k, v := range metadata {// 将元数据序列化为protobufprotoData, err := proto.Marshal(v)if err != nil {return errors.New("metadata to protobuf fail")}meta[k] = string(protoData)}// 组装多个写入操作txn.If(clientv3.Compare(clientv3.CreateRevision("meta_table"), "=", 0)).// 此处不允许相同的键重复写入Then(clientv3.OpPut("meta_table", meta["meta_table"]),clientv3.OpPut("meta_table1", meta["meta_table1"]),)// 提交事务resp, err := txn.Commit()if err != nil {// handle error!fmt.Printf("txn commit failed, err: %v\n", err)return nil}// 检查事务结果if !resp.Succeeded {fmt.Println("txn failed, some keys already exist")} else {fmt.Println("txn succeeded")}return nil
}func getMetadataFromEtcd(cli *clientv3.Client, metadataMap map[string]*message.Metadata, args ...interface{}) error {if len(args) == 0 {return errors.New("not key need read")}// 遍历key并从etcd中读取对应的key-valfor _, arg := range args {switch v := arg.(type) {case map[string]*message.Metadata: // 以map的形式指定key(即读取刚写入的数据){// map为空if len(v) == 0 {return errors.New("key map is nil")} else {for k, _ := range v {ctx, cancel := context.WithTimeout(context.Background(), 5*time.Second)meta, err := cli.Get(ctx, k)cancel()if err != nil {return errors.New("read matadata fail")}if len(meta.Kvs) == 0 {return errors.New("no key-value found for the given key")}// 把protobuf解析为Metadatavar retrievedMetadata message.Metadataerr = proto.Unmarshal(meta.Kvs[0].Value, &retrievedMetadata)if err != nil {return errors.New("protobuf to Metadata fail")}metadataMap[k] = &retrievedMetadata}}break}case string: // 单个读或批量读{if strings.Contains(v, "prefix") { // prefix读str := strings.Split(v, ":")key := str[0]// 创建事务txn := cli.Txn(context.Background())ctx, cancel := context.WithTimeout(context.Background(), 5*time.Second)meta, err := cli.Get(ctx, key, clientv3.WithPrefix())cancel()if err != nil {return errors.New("read matadata error")}if len(meta.Kvs) == 0 {return errors.New("no key-value found for the given key")}for _, m := range meta.Kvs {// 把protobuf解析为Metadatavar val message.Metadataerr = proto.Unmarshal(m.Value, &val)if err != nil {return errors.New("protobuf to Metadata fail")}metadataMap[string(m.Key)] = &val}// 提交事务resp, err := txn.Commit()if err != nil {return errors.New("txn commit failed")}// 检查事务结果if !resp.Succeeded {return errors.New("txn failed, some keys already exist")} else {return errors.New("txn succeeded")}} else { // 单条读ctx, cancel := context.WithTimeout(context.Background(), 5*time.Second)meta, err := cli.Get(ctx, v)cancel()if err != nil {return errors.New("read matadata error")}if len(meta.Kvs) == 0 {return errors.New("no key-value found for the given key")}var val message.Metadataerr = proto.Unmarshal(meta.Kvs[0].Value, &val)if err != nil {return errors.New("protobuf to Metadata fail")}metadataMap[string(meta.Kvs[0].Key)] = &val}break}default:return errors.New("args is unknown data type")}}return nil
}// 监听etcd上指定key的变化,如果发生变化则更新缓存中的元数据
func watchEtcdAndUpdate(cli *clientv3.Client, metadataMap map[string]*message.Metadata, key string) error {if key == "" {return errors.New("not key need watch")}ctx, cancel := context.WithCancel(context.Background())defer cancel()watch := cli.Watch(ctx, key, clientv3.WithPrefix(), clientv3.WithPrevKV())for w := range watch {for _, ev := range w.Events {fmt.Printf("Type: %s Key: %s Value: %s\n", ev.Type, ev.Kv.Key, ev.Kv.Value)// 根据变化修改缓存中的元数据switch ev.Type {case clientv3.EventTypePut:{var val message.Metadataerr := proto.Unmarshal(ev.Kv.Value, &val)if err != nil {return errors.New("protobuf to Metadata fail")}metadataMap[string(ev.Kv.Key)] = &valfor k, v := range metadataMap {fmt.Println("tableName:", k)fmt.Println("metadata ShardCount:", v.GetShardCount())}break}case clientv3.EventTypeDelete:{if ev.PrevKv.Key != nil {delete(metadataMap, string(ev.Kv.Key))}break}default:return errors.New("watch error")}}}return nil
}func main() {// 初始化一个缓存,模拟其它kingproxy节点metadataMap := make(map[string]*message.Metadata)// 连接etcd集群cli, err := connectToEtcd("120.92.144.250:2379")if err != nil {fmt.Println(err)return}defer cli.Close()// 初始化元数据metadata, err := initMetadata()if err != nil {fmt.Println(err)return}// 写入元数据//err = putMetadataToEtcdSingle(cli, metadata)//if err != nil {// fmt.Println(err)// return//}// 批量写入元数据err = putMetadataToEtcdBatch(cli, metadata)if err != nil {fmt.Println(err)return}// 读取元数据getMetadataFromEtcd(cli, metadataMap, "meta_table:prefix")// pre resultfor k, v := range metadataMap {fmt.Println("tableName:", k)fmt.Println("metadata ShardCount:", v.GetShardCount())}fmt.Println("--------------------------------------------------------------------------------")// watch key//watchEtcdAndUpdate(cli, metadataMap, "meta_table")return
}

完整 Go 工程

完整 Go 工程获取:https://download.csdn.net/download/weixin_47156401/89019290

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/766361.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

迷宫(一)(DFS BFS)

//新生训练 #include <bits/stdc.h> using namespace std; int n, m; bool f; char mp[15][15]; int vis[15][15]; int dir[4][2] {{-1, 0}, {0, 1}, {1, 0}, {0, -1}}; bool in(int x, int y) {return 0 < x && x < n && 0 < y && y …

2.1-如何配置华三网络设备的远程登录?(ssh,telnet,https)

&#xff08;1&#xff09;华三网络设备远程登录的配置实验 1.实验目标 目标&#xff1a;实现远程ssh登录基本配置(http,https登录&#xff09; 要求&#xff1a;配置后&#xff0c;即可使用远程工具进行远程登录。 配置过程&#xff1a; 1.创建交换机的本地帐号。&#xff0…

NFT交易市场-后端开发

首先我们需要配置好我们的ipfs&#xff0c;参考官方文档 1.https://docs.ipfs.tech/install/command-line/#system-requirementshttps://docs.ipfs.tech/how-to/command-line-quick-start/#initialize-the-repository 首先新建一个文件夹 然后在终端输入npm init -y命令进行初…

深入理解Redis的Sentinel机制

Sentinel简述 Sentinel为了解决什么问题&#xff1f; Sentinel&#xff08;哨岗、哨兵&#xff09;是Redis的高可用性&#xff08;high availability&#xff09;解决方案。 我们知道Redis 的主从复制模式可以将主节点的数据改变同步给从节点&#xff0c;这样从节点就可以起…

docker 和K8S知识分享

docker知识&#xff1a; 比如写了个项目&#xff0c;并且在本地调试没有任务问题&#xff0c;这时候你想在另外一台电脑或者服务器运行&#xff0c;那么你需要在另外一台电脑或者服务器配置相同的软件&#xff0c;比如数据库&#xff0c;web服务器&#xff0c;必要的插件和库等…

吴恩达机器学习笔记 二十七 决策树中连续值特征的选择 回归树

还是猫狗分类的案例&#xff0c;假如再增加一个特征weight&#xff0c;该值是一个连续的值&#xff0c;如何在决策树中使用该特征&#xff1f; 如下图所示&#xff0c;尝试不同的阈值&#xff0c;如 weight<9 , 此时左边有四个样本&#xff0c;都为猫&#xff0c;右边有六个…

分布式搜索引擎ES-RestClient查询文档快速入门

RestClient查询文档快速入门 文章目录 RestClient查询文档快速入门1.1、match_all1.2、全文检索查询1.3、精确查询1.4、复合查询-boolean query1.5、排序和分页1.6、高亮&#xff08;解析查询高亮结果&#xff09; 1.1、match_all package cn.mannor.hotel;import org.apache.…

C#,图论与图算法,计算图(Graph)的岛(Island)数量的算法与源程序

1 孤岛数 给定一个布尔矩阵,求孤岛数。一组相连的1形成一个岛。例如,下面的矩阵包含5个岛: 在讨论问题之前,让我们先了解什么是连接组件。无向图的连通分量是一个子图,其中每两个顶点通过一条路径相互连接,并且不与子图外的其他顶点连接。 所有顶点相互连接的图只有一个…

备战蓝桥杯(前缀和、差分篇)

Acwing 562.壁画 题目大意&#xff1a; 墙壁为一行&#xff0c; n n n个格子&#xff0c;每个时刻开始&#xff0c;先涂染料&#xff0c;再崩坏&#xff0c;涂过染料的格子不会崩坏&#xff0c;并且涂格子只能涂相邻未涂过的。 解题思路&#xff1a; 因为是先涂染料&#xff…

Tomcat 下载以及安装

Tomcat安装及配置教程主要分为四步&#xff1a; 步骤一&#xff1a;首先确认自己是否已经安装JDK 1. cmd&#xff1a;查看java的版本 步骤二&#xff1a;下载安装Tomcat 1. 下载tomcat :Apache Tomcat - Welcome! 2. 选择对应的tomcat版本&#xff1a; 3. 进行安装&#…

Uibot6.0 (RPA财务机器人师资培训第3天 )财务招聘信息抓取机器人案例实战

训练网站&#xff1a;泓江科技 (lessonplan.cn)https://laiye.lessonplan.cn/list/ec0f5080-e1de-11ee-a1d8-3f479df4d981https://laiye.lessonplan.cn/list/ec0f5080-e1de-11ee-a1d8-3f479df4d981https://laiye.lessonplan.cn/list/ec0f5080-e1de-11ee-a1d8-3f479df4d981(本博…

鸿蒙一次开发,多端部署(十四)一多开发实例(短信)

本章从系统预置的应用中&#xff0c;选择短信应用作为典型的案例&#xff0c;从页面开发和工程结构的角度&#xff0c;介绍"一多"的具体实践。系统的产品形态在不断丰富中&#xff0c;当前主要有默认设备和平板两种产品形态&#xff0c;本章的具体实践也将围绕这两种…

ALPHA开发板中CAN硬件图

一. 简介 前面文章学习了 IMX6ULL芯片的 CAN总线协议&#xff0c;CAN传输速率。 本文来搜索 ALPHA开发板中CAN硬件原理图&#xff0c;以及CAN设备节点信息。这里主要是CAN控制器的驱动&#xff0c;属于IMX6ULL芯片内部的驱动&#xff0c;NXP官方已经写好。 CAN控制器的驱动…

使用Python抓取抖音直播间数据的简易指南【第152篇—抓取数据】

&#x1f47d;发现宝藏 前些天发现了一个巨牛的人工智能学习网站&#xff0c;通俗易懂&#xff0c;风趣幽默&#xff0c;忍不住分享一下给大家。【点击进入巨牛的人工智能学习网站】。 使用Python抓取抖音直播间数据的简易指南 说明&#xff1a;本文已脱敏&#xff0c;隐去地址…

Keepalive与idle监测及性能优化

Keepalive 与 idle监测 Keepalive&#xff08;保活&#xff09;: Keepalive 是一种机制&#xff0c;通常用于TCP/IP网络。它的目的是确保连接双方都知道对方仍然存在并且连接是活动的。这是通过定期发送控制消息&#xff08;称为keepalive消息&#xff09;实现的。如果在预定时…

鸿蒙Harmony应用开发—ArkTS-if/else:条件渲染

ArkTS提供了渲染控制的能力。条件渲染可根据应用的不同状态&#xff0c;使用if、else和else if渲染对应状态下的UI内容。 说明&#xff1a; 从API version 9开始&#xff0c;该接口支持在ArkTS卡片中使用。 使用规则 支持if、else和else if语句。 if、else if后跟随的条件语句…

算法-双指针

目录 1、双指针遍历分割:避免开空间&#xff0c;原地处理 2、快慢指针&#xff1a;循环条件下的判断 3、左右指针&#xff08;对撞指针&#xff09;&#xff1a;分析具有单调性&#xff0c;避免重复计算 双指针又分为双指针遍历分割&#xff0c;快慢指针和左右指针 1、双指…

【leetcode热题】 位1的个数

编写一个函数&#xff0c;输入是一个无符号整数&#xff08;以二进制串的形式&#xff09;&#xff0c;返回其二进制表达式中数字位数为 1 的个数&#xff08;也被称为汉明重量&#xff09;。 提示&#xff1a; 请注意&#xff0c;在某些语言&#xff08;如 Java&#xff09;中…

Linux安装Nginx及配置TCP负载均衡

目录 1、安装编译工具及库文件2、下载解压Nginx压缩包3、Ngnix配置Tcp负载均衡4、配置Ngnix的文件5、Nginx启动 1、安装编译工具及库文件 yum -y install make zlib zlib-devel gcc-c libtool openssl openssl-devel pcre-devel2、下载解压Nginx压缩包 wget https://nginx.o…

5.7、【AI技术新纪元:Spring AI解码】Prompts

翻译: Spring AI 中的 Prompts 处理 Prompts 是指导 AI 模型生成特定输出的输入。这些 Prompts 的设计和措辞显著影响模型的响应。 在与 Spring AI 模型交互的最基础层面上,处理 Prompts 有点类似于在 Spring MVC 中管理“视图”。这涉及创建带有动态内容占位符的大量文本。…