一:连接到 etcd

package main

import (
"fmt"
"go.etcd.io/etcd/clientv3"
"time"
) //连接
func main() {
//客户端配置
config := clientv3.Config{
Endpoints: []string{"192.168.1.109:2379"},
DialTimeout: 5 * time.Second,
} //建立连接
if client, err := clientv3.New(config); err != nil {
fmt.Println(err)
return
} fmt.Println("connect success")
defer client.Close()
}

Endpoints:etcd的多个节点服务地址。

DialTimeout:创建client的首次连接超时时间,这里传了5秒,如果5秒都没有连接成功就会返回err;

一旦client创建成功,我们就不用再关心后续底层连接的状态了,client内部会重连。

二:KV操作:增 删 改 查

package main

import (
"context"
"fmt"
"go.etcd.io/etcd/clientv3"
"time"
) func main() {
client, err := clientv3.New(clientv3.Config{
Endpoints: []string{"192.168.1.109:2379", "192.168.1.109:2379"},
DialTimeout: 5 * time.Second,
})
if err != nil {
fmt.Println("connect failed, err :", err)
return
} defer client.Close() //控制超时
ctx, cancel := context.WithTimeout(context.Background(), time.Second)
//1. 增-存值
_, err = client.Put(ctx, "/demo/demo1_key", "demo1_value")
//操作完毕,cancel掉
cancel()
if err != nil {
fmt.Println("put failed, err:", err)
return
} //2. 查-获取值, 也设置超时
ctx, cancel = context.WithTimeout(context.Background(), time.Second)
resp, err := client.Get(ctx, "/demo/demo1_key")
// Get查询还可以增加WithPrefix选项,获取某个目录下的所有子元素
//eg: resp, err := client.Get(ctx, "/demo/", clientv3.WithPrefix())
cancel()
if err != nil {
fmt.Println("get failed err:", err)
return
} for _, item := range resp.Kvs { //Kvs 返回key的列表
fmt.Printf("%s : %s \n", item.Key, item.Value)
} //3. 改-修改值
ctx, _ = context.WithTimeout(context.Background(), time.Second)
resp, err := client.Put(ctx, "/demo/demo1_key", "update_value", clientv3.WithPrevKV()))
if err != nil {
fmt.Println("get failed err: ", err)
}
fmt.Println(string(resp.PrevKv.Value)) //4. 删-删除值
ctx, _ = context.WithTimeout(context.Background(), time.Second)
resp, err := client.Delete(ctx, "/demo/demo1_key")
if err != nil {
fmt.Println(err)
}
fmt.Println(resp.PrevKvs)
}

三:watch 监听

package main

import (
"context"
"fmt"
"go.etcd.io/etcd/clientv3"
"time"
) func main() {
client, err := clientv3.New(clientv3.Config{
Endpoints: []string{"192.168.1.109:2379"},
DialTimeout: time.Second,
})
if err != nil {
fmt.Println("connect failed err : ", err)
return
}
defer client.Close() client.Put(context.Background(), "/demo/demo2_key", "demo2_value")
go func() {
//watch
watchKey := client.Watch(context.Background(), "/demo/demo2_key")
for resp := range watchKey {
for _, item := range resp {
fmt.Printf("%s %q : %q \n", item.Type, item.Kv.key, item.Kv.Value)
}
}
} if resp, err := client.Put(context.TODO(), "/demo/demo2_key/", "demo2_watch"); err != nil {
fmt.Println(err)
} else {
fmt.Println(resp)
}
}

四:Transaction 事务

package main

import (
"context"
"fmt"
"go.etcd.io/etcd/clientv3"
"log"
"sync"
"time"
) func main() {
client, err := clientv3.New(clientv3.Config{
Endpoints: 5 * time.Second,
DialTimeout: 3 * time.Second,
})
if err != nil {
fmt.Println("connect failed err: ", err)
return
}
client.Close() var w sync.WaitGroup
w.Add(10)
key10 := "setnx"
for i := 0; i < 10; i++ {
go func(i int) {
time.Sleep(5 * time.Millisecond)
//通过key的Create_Revision 是否为 0 来判断key是否存在。其中If,Then 以及 Else 分支都可以包含多个操作。
//返回的数据包含一个successed字段,当为 true 时代表 If 为真
_, err := client.Txn(context.Background()).
If(clientv3.Compare(clientv3.CreateRevision(key10), "=", 0)).
Then(clientv3.OpPut(key10, fmt.Sprintf("%d", i))).
Commit()
if err != nil {
fmt.Println(err)
} w.Done()
}(i)
}
w.Wait() if resp, err := client.Get(context.TODO(), key10); err != nil {
log.Fatal(err)
} else {
log.Println(resp)
}
}

五:lease 租约

package main

import (
"time"
)
//From:https://github.com/the-gigi/go-etcd3-demo/blob/master/main.go
var (
dialTimeout = 2 * time.Second
requestTimeout = 10 * time.Second
) func main() {
ctx, _ := context.WithTimeout(context.Background(), requestTimeout)
client, err := clientv3.New(clientv3.Config{
DialTimeout: dialTimeout,
Endpoints: []string{"localhost:2379"},
}) if err != nil {
log.Fatal(err)
} kv := clientv3.NewKv(client) //Delete all keys
kv.Delete(ctx, "/demo/demo1_key", clientv3.WithPrefix()) gr, _ := kv.Get(ctx, "/demo/demo1_key")
if len(gr.Kvs) == 0 {
fmt.Println("no key")
} lease, err := client.Grant(ctx, 3)
if err != nil {
log.Fatal(err)
} //Insert key with a lease of 3 second TTL
kv.Put(ctx, "/demo/demo1_key", "demo1_value", clientv3.WithLease(lease.ID)) gr, _ = kv.Get(ctx, "/demo/demo1_key")
if len(gr.Kvs) == 1 {
fmt.Println("Found key")
} //let the TTL expire
time.Sleep(3 * time.Second) gr, _ = kv.Get(ctx, "/demo/demo1_key")
if len(gr.Kvs) == 0 {
fmt.Println("no more key")
}
}
//From:https://github.com/the-gigi/go-etcd3-demo/blob/master/main.go

etcd 使用: golang 例子的更多相关文章

  1. 一个Golang例子:for + goroutine + channel

    Rob Pike 在 Google I/O 2012 - Go Concurrency Patterns 里演示了一个例子(daisy chain). 视频地址:https://www.youtube ...

  2. ETCD分布式锁实现选主机制(Golang实现)

    ETCD分布式锁实现选主机制(Golang) 为什么要写这篇文章 做架构的时候,涉及到系统的一个功能,有一个服务必须在指定的节点执行,并且需要有个节点来做任务分发,想了半天,那就搞个主节点做这事呗,所 ...

  3. ETCD相关介绍--整体概念及原理方面

    etcd作为一个受到ZooKeeper与doozer启发而催生的项目,除了拥有与之类似的功能外,更专注于以下四点. 简单:基于HTTP+JSON的API让你用curl就可以轻松使用. 安全:可选SSL ...

  4. GO开发:etcd用法

    etcd是什么? A highly-available key value store for shared configuration and service discovery.是一个键值存储仓库 ...

  5. 万级K8s集群背后etcd稳定性及性能优化实践

    背景与挑战 随着腾讯自研上云及公有云用户的迅速增长,一方面,腾讯云容器服务TKE服务数量和核数大幅增长, 另一方面我们提供的容器服务类型(TKE托管及独立集群.EKS弹性集群.edge边缘计算集群.m ...

  6. 万级K8s集群背后 etcd 稳定性及性能优化实践

    1背景与挑战随着腾讯自研上云及公有云用户的迅速增长,一方面,腾讯云容器服务TKE服务数量和核数大幅增长, 另一方面我们提供的容器服务类型(TKE托管及独立集群.EKS弹性集群.edge边缘计算集群.m ...

  7. Go实现海量日志收集系统(三)

    再次整理了一下这个日志收集系统的框,如下图 这次要实现的代码的整体逻辑为: 完整代码地址为: https://github.com/pythonsite/logagent etcd介绍 高可用的分布式 ...

  8. session.go

    package {             so.ttl = ttl         }     } } // WithContext assigns a context to the session ...

  9. syncer.go

    package ) ) ) ].Key,)) )) }

随机推荐

  1. JAVA内存存储分配粗略讲解

    以String类型为例:String s1 = "ABC"; String s2 = "ABC"; String s3 = new String("A ...

  2. FutureWarning: get_value is deprecated and will be removed in a future release. Please use .at[] or .iat[] accessors instead print(labels_df.get_value(patients,col=1))

    这是因为pandas的版本高了,0.21之后就已经将这个方法干掉了.直接装成0.20之前的就好

  3. java 内部类(简单使用)

    什么是内部类 1.内部类是指在一个外部类的内部再定义一个类. 2.内部类作为外部类的一个成员,依附于外部类而存在. 3.内部类可为静态,可用protected和private修饰(而外部类只能使用pu ...

  4. Python3之Requests模块详解

    # 导入 Request模块 # 若本机无自带Request模块,可自行下载或者使用pip进行安装 # python版本Python3 import requests import json #### ...

  5. 【JDK】MacBook 安装JDK及卸载步骤

    一.安装步骤 1.官网下载jdk https://www.oracle.com/technetwork/java/javase/downloads/index.html 勾选   Accept Lic ...

  6. linux安装locust及遇到的坑

    安装locust很简单,简单的我以为pip install一下就好了,结果运行locust死活不行,后来的后来,,才搞清楚还要进行以下操作. 前提: 1.已安装Python3和pip3(或者都是2,版 ...

  7. 油猴Tampermonkey离线安装流程(附文件)

    1.下载插件插件包,然后解压(解压到你想放插件的位置,其实任意位置都可以,记住解压的位置) 链接:https://pan.baidu.com/s/1aanhsb6ZlapnzBeBRtp3Hg 提取码 ...

  8. 13. Ajax技术

    在传统的Web应用模式中,页面中用户的每一次操作都将触发一次返回Web服务器的HTTP请求,服务器进行相应的处理后,返回一个HTML页面的客户端.而在Ajax应用中,页面中的用户的操作将通过Ajax引 ...

  9. JAVA中java.lang.OutOfMemoryError常见的解决方式

    在开发中我们很多人都遇到过内存溢出的情况,其实内存溢出分几种形式: 1.tomcat中java.lang.OutOfMemoryError: PermGen space异常处理(最常见的) 概念大家可 ...

  10. hbase数据导入

    hbase数据导入: 参考http://blog.csdn.net/hua840812/article/details/7414875,在把代码copy下来后,发现运行总是报错: java.io.IO ...