【Go】微服务架构下实现etcd服务注册与服务发现

中心网关:gateway
四个微服务:user、message、note、relationship
在这里插入图片描述

1 中心网关实现服务发现

1.1 设计EtcdDiscovery类

package entityimport ("context""fmt"clientv3 "go.etcd.io/etcd/client/v3""gonote/gateway/internal/option"messageService "gonote/message/service"noteService "gonote/note/service"relationshipService "gonote/relationship/service"userService "gonote/user/service""google.golang.org/grpc""google.golang.org/grpc/credentials/insecure""sync""time"
)type EtcdDiscovery struct {etcdClient *clientv3.ClientserviceMap map[string]interface{}mu         sync.RWMutex
}func NewEtcdDiscovery(ip string, port int) (*EtcdDiscovery, error) {etcdCli, err := clientv3.New(clientv3.Config{Endpoints:   []string{fmt.Sprintf("%v:%v", ip, port)},DialTimeout: time.Second * 3,})if err != nil {return nil, err}return &EtcdDiscovery{etcdClient: etcdCli,serviceMap: make(map[string]interface{}),}, nil
}func (ed *EtcdDiscovery) Start(serviceNames []string) {for _, serviceName := range serviceNames {resp, err := ed.etcdClient.Get(context.TODO(), serviceName)if err != nil {panic(err)}addr := string(resp.Kvs[0].Value)conn, err := grpc.Dial(addr,grpc.WithTransportCredentials(insecure.NewCredentials()))if err != nil {panic(err)}// etcd存储的是各个微服务的监听地址,通过监听地址创建服务实例switch serviceName {case option.User:grpcCli := userService.NewUserServiceClient(conn)ed.serviceMap[serviceName] = grpcClicase option.Relationship:grpcCli := relationshipService.NewRelationshipServiceClient(conn)ed.serviceMap[serviceName] = grpcClicase option.Note:grpcCli := noteService.NewNoteServiceClient(conn)ed.serviceMap[serviceName] = grpcClicase option.Message:grpcCli := messageService.NewMessageServiceClient(conn)ed.serviceMap[serviceName] = grpcCli}// 开启协程,监听etcd的变化,动态维护各个grpc服务实例go ed.watch(serviceName)}
}func (ed *EtcdDiscovery) watch(serviceName string) {watchChan := ed.etcdClient.Watch(context.TODO(), serviceName)for event := range watchChan {for _, e := range event.Events {if e.Type == clientv3.EventTypePut {addr := string(e.Kv.Value)conn, err := grpc.Dial(addr,grpc.WithTransportCredentials(insecure.NewCredentials()))if err != nil {continue}ed.mu.Lock()switch serviceName {case option.User:grpcCli := userService.NewUserServiceClient(conn)ed.serviceMap[serviceName] = grpcClicase option.Relationship:grpcCli := relationshipService.NewRelationshipServiceClient(conn)ed.serviceMap[serviceName] = grpcClicase option.Note:grpcCli := noteService.NewNoteServiceClient(conn)ed.serviceMap[serviceName] = grpcClicase option.Message:grpcCli := messageService.NewMessageServiceClient(conn)ed.serviceMap[serviceName] = grpcCli}ed.mu.Unlock()} else if e.Type == clientv3.EventTypeDelete {ed.mu.Lock()delete(ed.serviceMap, serviceName)ed.mu.Unlock()}}}
}func (ed *EtcdDiscovery) GetServiceAddr(serviceName string) interface{} {ed.mu.RLock()defer ed.mu.RUnlock()return ed.serviceMap[serviceName]
}

1.2 在web启动时,初始化EtcdDiscovery

package mainimport ("gonote/gateway/internal""gonote/gateway/internal/option""gonote/gateway/internal/util"
)func init() {util.InitLogger()util.InitWebSocketServer(64)err := util.InitRedis()if err != nil {panic(err)}util.InitKafka(option.Topic)util.InitEtcdDiscovery([]string{option.User,option.Relationship,option.Note,option.Message})
}func main() {engine := internal.Router()if err := engine.Run("0.0.0.0:9090"); err != nil {panic(err)}
}

1.3 调用EtcdDiscovery获取服务实例

举个用户注册的例子:

func UserLogin(c *gin.Context) {em := c.PostForm("email")pwd := c.PostForm("pwd")// 获取服务实例cli := util.EtcdDiscovery.GetServiceAddr(option.User).(service.UserServiceClient)// 调用服务_, err := cli.UserLogin(context.TODO(), &service.User{Email: em,Pwd:   pwd,})if err != nil {c.JSON(200, gin.H{"code": 1,"msg":  err.Error(),})return}// 生成jwt令牌token, err := util.GenToken(em)if err != nil {c.JSON(200, gin.H{"code": 1,"msg":  err.Error(),})return}// session维护长连接session := sessions.Default(c)session.Set("email", em)err = session.Save()if err != nil {c.JSON(200, gin.H{"code": 1,"msg":  err.Error(),})return}c.JSON(200, gin.H{"code": 0,"data": token,})
}

2 微服务端进行服务注册

user业务对应的微服务:

func init() {util.InitLogger()err := util.InitDB()if err != nil {panic(err)}util.InitKafka(option.Topic)util.InitEtcdCli()
}func main() {addr := option.IP + ":" + option.Portln, err := net.Listen("tcp", addr)if err != nil {panic(err)}defer ln.Close()defer util.EtcdCli.Close()defer util.KafkaCli.Close()// 服务注册_, err = util.EtcdCli.Put(context.TODO(), "user", addr)if err != nil {panic(err)}server := grpc.NewServer()service.RegisterUserServiceServer(server, &service.UserServiceImpl{})if err = server.Serve(ln); err != nil {panic(err)}
}

通过etcd命令查看相关注册信息
在这里插入图片描述

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

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

相关文章

重置vCenter的root和administrator@vsphere.local密码

1:首先要重置root密码,登录vCenter安装的ESXI主机,重启vCenter。 2::重启机器的出现下面界面的时候按e键。 3:按e后出现下面的界面。 4:在最后一行结尾处输入rw init/bin/bash ,之后按ctrl-x或者F10重启。 …

spring boot 使用 Kafka

一、Kafka作为消息队列的好处 高吞吐量:Kafka能够处理大规模的数据流,并支持高吞吐量的消息传输。 持久性:Kafka将消息持久化到磁盘上,保证了消息不会因为系统故障而丢失。 分布式:Kafka是一个分布式系统&#xff0c…

pgsql中in 和 join 怎们选

前提: 两个表只有一个关联字段 IN的使用: 1、适用场景:当需要关联的数据量较小时(例如几十到几百条),使用IN是可以接受的。IN子句通常用于WHERE条件中,以指定某列的值是否在给定列表中。 2、性…

7000字详解Spring Boot项目集成RabbitMQ实战以及坑点分析

本文给大家介绍一下在 Spring Boot 项目中如何集成消息队列 RabbitMQ,包含对 RibbitMQ 的架构介绍、应用场景、坑点解析以及代码实战。 我将使用 waynboot-mall 项目作为代码讲解,项目地址:https://github.com/wayn111/waynboot-mall。本文大…

MATLAB - 仿真单摆的周期性摆动

系列文章目录 前言 本例演示如何使用 Symbolic Math Toolbox™ 模拟单摆的运动。推导摆的运动方程,然后对小角度进行分析求解,对任意角度进行数值求解。 一、步骤 1:推导运动方程 摆是一个遵循微分方程的简单机械系统。摆最初静止在垂直位置…

大数据 - Hadoop系列《四》- MapReduce(分布式计算引擎)的核心思想

上一篇: 大数据 - Hadoop系列《三》- MapReduce(分布式计算引擎)概述-CSDN博客 目录 13.1 MapReduce实例进程 13.2 阶段组成 13.4 概述 13.4.1 🥙Map阶段(映射) 13.4.2 🥙Reduce阶段执行过…

【Spark系列6】如何做SQL查询优化和执行计划分析

Apache Spark SQL 使用 Catalyst 优化器来生成逻辑执行计划和物理执行计划。逻辑执行计划描述了逻辑上如何执行查询,而物理执行计划则是 Spark 实际执行的步骤。 一、查询优化 示例 1:过滤提前 未优化的查询 val salesData spark.read.parquet(&quo…

Vue2:请求接口的两种方式axios和vue-resource

一、场景描述 前端和后端的交互,肯定是要发生接口调用的 这个时候,就要涉及前端如何向后端接口发送请求,获取数据 二、请求方式 1、axios方式(推荐) 这个方式本质就是ajax,底层就是对xhr(XMLHttpRequest)的封装 1、安装axios…

STM32F407移植OpenHarmony笔记3

接上一篇,搭建完环境,找个DEMO能跑,现在我准备尝试从0开始搬砖。 首先把/device和/vendor之前的代码全删除,这个时候用hb set命令看不到任何项目了。 /device目录是硬件设备目录,包括soc芯片厂商和board板级支持代码…

JAVA线程执行中断方式和ElasticSearch未捕获异常的处理方式

JAVA线程执行中断方式 Java中只能通过协作的方式取消 第一种是通过标志位实现,假设有个计算所有素数的任务,每次计算前检查下是否取消的标志位,如果为true则退出计算。调用方想要取消任务的话,则将标志位设为true。但这种方法无法…

Linux 系统相关的命令

参考资料 Linux之chmod使用【linux】chmod命令详细用法 目录 一. 系统用户相关1.1 查看当前访问的主机和用户1.2 切换用户1.2.1 设置root用户密码1.2.2 普通用户和root用户切换 1.4 系统状态1.4.1 vmstat 查看当前系统的状态1.4.2 history 查看系统中输入过的命令 二. 系统文件…

React18-列表数据实现用户删除、批量删除

用户删除、批量删除接口 删除、批量删除接口 接口地址 POST/users/delete请求参数 {userIds: [] }参数为数组,删除和批量删除共用 功能介绍 单个删除 删除按钮绑定事件,点击显示弹框确认。 // 删除 const handleDel (values: DataType) > {//…

图扑 HT UI 5.0 全新升级,开箱即用!

为顺应数字时代的不断发展,图扑 HT UI 5.0 在原有功能强大的界面组件库的基础上进行了全面升级,融入了更先进的技术、创新的设计理念以及更加智能的功能。HT UI 5.0 使用户体验更为直观、个性化,并在性能、稳定性和安全性等方面达到新的高度。…

githacker安装详细教程,linux添加环境变量详细教程(见标题三)

笔者是ctf小白,这两天也是遇到.git泄露的题目,需要工具来解决问题,在下载和使用的过程中也是遇到很多问题,写此篇记录经验,以供学习 在本篇标题三中有详细介绍了Linux系统添加环境变量的操作教程,以供学习 …

TensorFlow2实战-系列教程6:猫狗识别3------迁移学习

🧡💛💚TensorFlow2实战-系列教程 总目录 有任何问题欢迎在下面留言 本篇文章的代码运行界面均在Jupyter Notebook中进行 本篇文章配套的代码资源已经上传 猫狗识别1 数据增强 猫狗识别2------数据增强 猫狗识别3------迁移学习 1、迁移学习 …

TensorFlow2实战-系列教程15:Resnet实战3

🧡💛💚TensorFlow2实战-系列教程 总目录 有任何问题欢迎在下面留言 本篇文章的代码运行界面均在Jupyter Notebook中进行 本篇文章配套的代码资源已经上传 Resnet实战1 Resnet实战2 Resnet实战3 7、训练脚本train.py解读------配置训练参数 …

解读4篇混合类型文件Polyglot相关的论文

0. 引入 Polyglot文件指的是混合类型文件,关于混合类型文件的基础,请参考文末给出的第一个链接(参考1)。 1. Toward the Detection of Polyglot Files 1.1 主题 这篇2022年的论文,提出了Polyglot文件的检测方法。虽…

C++核心编程:文件操作 笔记

5.文件操作 程序运行时产生的数据都属于临时数据&#xff0c;程序一旦允许结束都会被释放。通过文件可以将数据持久化 C中对文件操作需要包含头文件<fstream> 文件类型分为两种&#xff1a; 文本文件 - 文件以文本的ASCII码形式存储在计算机中二进制文件 - 文件以文本…

openssl3.2 - .pod文件的查看方法

文章目录 .pod文件的查看方法概述笔记初步的解决方法备注 - pod2html.bat的详细用法好像Perl就自带这个BATEND .pod文件的查看方法 概述 看到openssl源码目录下有很多.pod文件, 软件发布的帮助内容都在里面. 当make install后, 大部分的.pod都会转成html文件, 但是有一部分…

【Java程序设计】【C00215】基于SSM的勤工助学管理系统(论文+PPT)

基于SSM的勤工助学管理系统&#xff08;论文PPT&#xff09; 项目简介项目获取开发环境项目技术运行截图 项目简介 这个一个基于SSM的勤工助学管理系统&#xff0c;本系统共分为三种权限&#xff1a;管理员、教师和学生 管理员&#xff1a;首页、个人中心、教师管理、学生管理…