为什么?

为什么会有服务注册和服务发现?在它以前我们是怎么做的?

举个例子:

比如我们做MySQL读写分离,就在本地配置一个文件,然后程序读取这个配置文件里的数据进行数据库读写分离的设置。

但是随着业务发展迅速,业务模块越来越多,数据也越来越多,MySQL数据库也越来越多,需要读取MySQL服务的业务模块也越来越多。每次增加MySQL实例,每个业务模块都要手动去写一次本地配置。

想一想这里有什么问题?可以改进吗?

每次手动写配置,是不是很浪费时间,能不能做到自动一些?能不能做到只配置一次,业务模块就可以自动读取更新后的配置?于是我们就开始动脑筋了,就开始琢磨解决的办法了。

在计算机领域,没有什么是增加一个中间层解决不了的问题,有的话,“那就在增加一层”,对,最后那句是我说的- -!。

于是就想:能不能把所有的配置集中存放,程序模块的配置更新由各个程序自动的来配置中心读取,更新配置。而不是每次手动的增加每个程序模块的配置。

于是我们就增加了一个“中间层”- 配置中心,集中存储,管理分散在各个程序中的配置。配置中心就向外面提供配置服务。

不过Etcd应该属于服务配置中心,属于更加细化的配置中心。

还有更加专注的配置中心,比如Apollo,Nacos,disconf等等。

是什么?

什么是服务注册和服务发现?

  • 服务注册 :就是向配置中心注册一个配置的功能。
  • 服务发现:就是发现配置中心增加、修改、删除了配置的功能。发现配置的异动情况,让更新配置。

会遇到哪些问题?

服务发现和服务注册怎么解决问题以及会遇到哪些问题?

  1. 怎么标识一个服务?

    在计算机里,标识一个计算机中运行的程序可以用ip+端口形式来标识,配置服务也可以用这种方式。
  2. 怎么发现一个配置服务的异动?

    配置更新,删除(下线),增加了,怎么发现配置的这些异动情况?一种是长轮询,一种是心跳,每隔多长时间检查一次,还有发布订阅模式。
  3. 服务是否可用?

    这种就要检查服务的可用情况了,一般是健康检查。

... ...

等等问题

怎么解决

业内主要的几个解决方案ZooKeeper,Etcd,Consul ,等。我们介绍etcd的使用。

etcd介绍

etcd是CoreOS团队于2013年6月发起的开源项目,它的目标是构建一个高可用的分布式键值(key-value)数据库。etcd内部采用raft协议作为一致性算法,etcd基于Go语言实现。

前面博客文章也有一些它的使用情况。

它是一个kv键值的数据库:

所以可以存储数据,那么就可以存储配置数据。

它有watch功能:

所以可以发现配置的异动情况。变化时候可以起到通知作用。

它有key TTL功能:

etcdv3可以通过lease租约,设置服务生存周期TTL,让key自动过期,通过KeepAlive定期续租,避免过期。

当然,它还可以做分布式锁、分布式队列等等其他功能,你可以google去学习它的其他功能。

例子

  • 注册服务和获取服务
// PutValueDemo 插入值demo,配置里面put相当于注册一个服务(服务注册),get相当于获取一个服务(服务发现)
func (demo EtcdDemo) PutValueDemo(client *clientv3.Client) {
fmt.Println("\n...put value demo...")
fmt.Println("put value: ", demo)
_, err := demo.PutValue(client)
if err != nil {
fmt.Println("failed to put value, err: ", err)
return
} getVal, err := demo.GetValue(client)
if err != nil {
fmt.Println("get val err: ", err)
return
}
fmt.Println("get key:",string(getVal.Kvs[0].Key),",value:", string(getVal.Kvs[0].Value), ", Revision: ", getVal.Header.Revision)
}

如果你单独在main函数里测试PutValueDemo(),

client, err := clientv3.New(clientv3.Config{
Endpoints: []string{"127.0.0.1:2379"},
DialTimeout: time.Second * 5,
})
if err != nil {
fmt.Println("failed to connect etcd: ", err)
return
}
defer client.Close() demo := EtcdDemo{Key: "test1", Val: "val1"}
demo.PutValueDemo(client)

你会发现:

Revision 的值,每次都是增加的。

get val: val1 ,key: test1 , Revision:  1

get val: val1 ,key: test1 , Revision:  2

这也是etcd的一个特性,可以用个来做分布式锁。

  • watch 监测服务变化
// watchDemo 监听key的变化
func (demo EtcdDemo) WatchDemo(client *clientv3.Client) {
fmt.Println("\n...watch demo...")
stopChan := make(chan interface{}) // 是否停止信号
go func() {
watchChan := client.Watch(context.TODO(), demo.Key, clientv3.WithPrefix())
for {
select {
case result := <- watchChan:
for _, event := range result.Events {
fmt.Printf("%s %q : %q\n", event.Type, event.Kv.Key, event.Kv.Value)
}
case <-stopChan:
fmt.Println("stop watching...")
return
}
}
}() for i := 0; i < 5; i++ {
var demo EtcdDemo
demo.Key = fmt.Sprintf("key_%02d", i)
demo.Val = strconv.Itoa(i)
demo.PutValue(client)
}
time.Sleep(time.Second * 1) stopChan <- 1 //停止watch,在插入就不会监听到了
}

运行后:

...watch demo...

putResp Revision:  13

PUT "key_00" : "0"

PUT "key_01" : "1"

putResp Revision:  14

PUT "key_02" : "2"

putResp Revision:  15

PUT "key_03" : "3"

putResp Revision:  16

PUT "key_04" : "4"

putResp Revision:  17

stop watching...

Revision单调递增变化的。

watch 可以监测到put变化的情况。

  • 租约lease
// LeaseDemo 租约
func (demo EtcdDemo) LeaseDemo(client *clientv3.Client) {
fmt.Println("\n...lease demo...") lease, err := client.Grant(context.TODO(), 2) //创建一个租约
if err != nil {
fmt.Println("grant err: ", err)
return
} testKey := "testleasekey"
// 给这个testkey一个 2秒的TTL租约
client.Put(context.TODO(), testKey, "testvalue", clientv3.WithLease(lease.ID))
getVal, err := client.Get(context.TODO(), testKey)
if err != nil {
fmt.Println("get val err: ", err)
return
}
vallen := len(getVal.Kvs)
fmt.Println("before time sleep, val len: ", vallen) fmt.Println("sleep 4 seconds")
time.Sleep(4 * time.Second) //睡眠4秒,让租约过期 getVal, _ = client.Get(context.TODO(), testKey)
vallen = len(getVal.Kvs)
fmt.Println("after 4 seconds, val len: ", vallen)
}

运行:

...lease demo...

before time sleep, val len:  1

sleep 4 seconds

after 4 seconds, val len:  0 //这里租约到期,删掉了这个key和val

完整代码在这里github

参考

golang微服务实践:服务注册与服务发现 - Etcd的使用的更多相关文章

  1. .Net微服务实践(五)[服务发现]:Consul介绍和环境搭建

    目录 介绍 服务发现 健康检查.键值存储和数据中心 架构 Consul模式 环境安装 HTTP API 和Command CLI 示例API介绍 最后 在上篇.Net微服务实践(四)[网关]:Ocel ...

  2. .Net微服务实践(四)[网关]:Ocelot限流熔断、缓存以及负载均衡

    目录 限流 熔断 缓存 Header转化 HTTP方法转换 负载均衡 注入/重写中间件 后台管理 最后 在上篇.Net微服务实践(三)[网关]:Ocelot配置路由和请求聚合中我们介绍了Ocelot的 ...

  3. Golang微服务实践

    背景 在之前的文章<漫谈微服务>我已经简单的介绍过微服务,微服务特性是轻量级跨平台和跨语言的服务,也列举了比较了集中微服务通信的手段的利弊,本文将通过RPC通信的方式实现一个增删查Redi ...

  4. Node.js微服务实践(一)

    什么是微服务 微服务是一种架构风格,一个大型复杂软件应用由一个或多个微服务组成.系统中的各个微服务可被独立部署,各个微服务之间是松耦合的.每个微服务仅关注于完成一件任务并很好地完成该任务.在所有情况下 ...

  5. 【GoLang】golang 微服务框架 介绍

    原文如下: rpcx是一个类似阿里巴巴 Dubbo 和微博 Motan 的分布式的RPC服务框架,基于Golang net/rpc实现. 谈起分布式的RPC框架,比较出名的是阿里巴巴的dubbo,包括 ...

  6. 【GoLang】GoLang 微服务、开源库等参考资料

    参考资料: GoLang书籍: https://github.com/dariubs/GoBooksGo名库: https://github.com/Unknwon/go-rock-libraries ...

  7. ASP.NET Core OceLot 微服务实践

    1.OceLot中间件介绍 在传统的BS应用中,随着业务需求的快速发展变化,需求不断增长,迫切需要一种更加快速高效的软件交付方式.微服务可以弥补单体应用不足,是一种更加快速高效软件架构风格.单体应用被 ...

  8. QCon技术干货:个推基于Docker和Kubernetes的微服务实践

    2016年伊始,Docker无比兴盛,如今Kubernetes万人瞩目.在这个无比需要创新与速度的时代,由容器.微服务.DevOps构成的云原生席卷整个IT界.在近期举办的QCon全球软件开发大会上, ...

  9. 基于SpringCloud的微服务实践

    微服务不同于单一架构应用, 是典型的分布式场景, 各服务之间通过IPC进行通信. 实现微服务的过程中, 我们需要解决以下问题: 服务注册和服务发现. 根据应用选择合适的通信协议和数据协议. 例如可以选 ...

  10. .NET CORE微服务实践

    .NET CORE微服务实践 https://www.cnblogs.com/zengqinglei/p/9570343.html .NET CORE 实践部署架构图 实践源码:https://git ...

随机推荐

  1. [转帖]Load Base Split

    https://docs.pingcap.com/zh/tidb/stable/configure-load-base-split#load-base-split Load Base Split 是 ...

  2. ARM下KVM虚拟化的损耗验证--redis

    ARM下KVM虚拟化的损耗验证 摘要 看Windows 上面的 Workstation的虚拟机的 网络层的延迟特别高. 突然想之前统计都是直接在本地验证的, 只考虑了虚拟化CPU的性能损耗 没有考虑虚 ...

  3. [转]流程自动化机器人(RPA)概念、原理与实践

    [转]流程自动化机器人(RPA)概念.原理与实践 http://blog.sina.com.cn/s/blog_be0833d00102yho9.html 大多数人每天都会使用到一些机器人流程自动化工 ...

  4. 【转贴】2019.3 学习向SP打造指南 篇一:微软神器Surface产品线全系列详细介绍

    学习向SP打造指南 篇一:微软神器Surface产品线全系列详细介绍 2019-03-01 22:30:00 161点赞 699收藏 141评论 https://post.smzdm.com/p/a5 ...

  5. 查看java所有的线程信息

    最近一直有一个困惑, 不知道如何查看所有的java的线程信息. 今天看blog时发现了一个简单方法 ps -Tp $pid 就可以了 也可以使用 ps- Lfp $pid的方式 这里简单写一下统计方法 ...

  6. vue过滤器(filter)的使用

    过滤器分全局过滤器和局部过滤器 <div id="app"> <p>电脑价格:{{price | addPriceIcon}}</p> < ...

  7. 动态添加input,然后获取所有的input框中的值

    今天遇见一个问题. 点击按钮,动态添加input框(可以添加多个) 然后搜集用户在input中输入的值. 我刚刚在纠结,给input框中注入事件. 但是这样会很麻烦. 经过同事的指点. 我直接去拿v- ...

  8. ActiveReports报表行号

    =RunningValue(Fields!字段名称.Value, CountDistinct, "矩表分组名称") RunningValue(Fields!区域.Value, Co ...

  9. 从零开始配置 vim(11)——插件管理

    之前我们介绍了基础配置部分和快捷键配置部分.如果你配置了这两个部分,vim已经算是比较好用了.但是作为代码编辑器来讲还是显的比较简陋,用这些配置来完成日常的编码任务会显得力不从心.vim比较强大的一点 ...

  10. vim 从嫌弃到依赖(5)——普通模式的一些操作

    通过前面几章内容的铺垫,基本已经介绍完了普通模式的大部分内容,按照进度下面会依次介绍插入模式.命令模式.选择模式的一些操作.根据不同模式提供功能的多少和使用频率,篇幅会有长有短.本来这篇文章应该介绍插 ...