.NET Core + Consul 服务注册与发现

在分布式架构中,服务治理是必须面对的问题,如果缺乏简单有效治理方案,各服务之间只能通过人肉配置的方式进行服务关系管理,当遇到服务关系变化时,就会变得极其麻烦且容易出错。

Consul[1] 是一个用来实现分布式系统服务发现与配置的开源工具。它内置了服务注册与发现框架、分布一致性协议实现、健康检查、Key/Value存储、多数据中心方案,不再需要依赖其他工具(比如 ZooKeeper 等),使用起来也较为简单。

Consul 架构

Consul 集群支持多数据中心,在上图中有两个 DataCenter,他们通过 Internet 互联,为了提高通信效率,只有 Server 节点才加入跨数据中心的通信。在单个数据中心中,Consul 分为 Client 和 Server 两种节点(所有的节点也被称为 Agent),Server 节点保存数据,Client 负责健康检查及转发数据请求到 Server,本身不保存注册信息;Server 节点有一个 Leader 和多个 Follower,Leader 节点会将数据同步到 Follower,Server 节点的数量推荐是3个或者5个,在 Leader 挂掉的时候会启动选举机制产生一个新 Leader。

Consul 集群搭建

这里使用 Docker 搭建 3个 Server 节点 + 1 个 Client 节点,API 服务通过 Client 节点进行服务注册和发现。

从 Docker Hub 拉取 Consul 镜像

docker pull consul

启动 3个 Server 节点 + 1 个 Client 节点

docker-compose.yaml 如下:

version: '3'services:cs1:image: consulcommand: agent -server -client=0.0.0.0 -bootstrap-expect=3 -node=cs1 -data-dir=/datavolumes:- /usr/local/docker/consul/data/cs1:/datacs2:image: consulcommand: agent -server -client=0.0.0.0 -retry-join=cs1 -node=cs2 -data-dir=/datavolumes:- /usr/local/docker/consul/data/cs2:/datadepends_on:- cs1cs3:image: consulcommand: agent -server -client=0.0.0.0 -retry-join=cs1 -node=cs3 -data-dir=/datavolumes:- /usr/local/docker/consul/data/cs3:/datadepends_on:- cs1cc1:image: consulcommand: agent -client=0.0.0.0 -retry-join=cs1 -ui -node=cc1 -data-dir=/dataports:- 8500:8500volumes:- /usr/local/docker/consul/data/cc1:/datadepends_on:- cs2- cs3

主要参数说明:

参数名解释
-server设置为 Server 类型节点,不加则为 Client 类型节点
-client注册或者查询等一系列客户端对它操作的IP,默认是127.0.0.1
-bootstrap-expect集群期望的 Server 节点数,只有达到这个值才会选举 Leader
-node指定节点名称
-data-dir数据存放位置
-retry-join指定要加入的节点地址(组建集群)
-ui启用 UI 界面

集群状态

e002ca62ac24  为容器名称,可通过 docker ps | grep consul 查看,选择任意一个即可

查看节点状态和类型
docker exec -t e002ca62ac24 consul members

当前为3 个 Server 类型节点 ,1 个 Client 类型节点。

查看 Server 节点类型
docker exec -t e002ca62ac24 consul operator raft list-peers

当前为 cs1 为 leader,可以测试将 cs1 stop 观察 leader 的重新选举。

通过 http://192.168.124.9:8500 UI 界面查看 Consul 节点状态如下:(192.168.124.9 是 consul 容器外部访问 IP)

.NET Core 接入 Consul

  1. 创建 .NET Core WebAPI(3.1) 服务 ServiceA(2个实例) 和 ServiceB

  2. NuGet 安装 Consul

  3. 注册到 Consul 的核心代码如下(源码下载[2]):

    public static class ConsulBuilderExtensions
    {public static IApplicationBuilder RegisterConsul(this IApplicationBuilder app, IHostApplicationLifetime lifetime, ConsulOption consulOption){var consulClient = new ConsulClient(x =>{x.Address = new Uri(consulOption.Address);});var registration = new AgentServiceRegistration(){ID = Guid.NewGuid().ToString(),Name = consulOption.ServiceName,// 服务名Address = consulOption.ServiceIP, // 服务绑定IPPort = consulOption.ServicePort, // 服务绑定端口Check = new AgentServiceCheck(){DeregisterCriticalServiceAfter = TimeSpan.FromSeconds(5),//服务启动多久后注册Interval = TimeSpan.FromSeconds(10),//健康检查时间间隔HTTP = consulOption.ServiceHealthCheck,//健康检查地址Timeout = TimeSpan.FromSeconds(5)}};// 服务注册consulClient.Agent.ServiceRegister(registration).Wait();// 应用程序终止时,服务取消注册lifetime.ApplicationStopping.Register(() =>{consulClient.Agent.ServiceDeregister(registration.ID).Wait();});return app;}
    }
    
  4. 添加配置如下:

    "Consul": {"ServiceName": "service-a","ServiceIP": "192.168.124.11",   // 当前服务访问 IP"ServicePort": 8000,"ServiceHealthCheck": "http://192.168.124.11:8000/healthCheck","Address": "http://192.168.124.9:8500"
    }
    
  5. 注册成功结果如下:

  6. ServiceB 调用 ServiceA 接口

    ServiceB 通过 ConsulClient 进行服务发现,获取到 ServiceA 的地址,然后随机请求请求任意一个实例,关键代码如下:

    using (var consulClient = new ConsulClient(a => a.Address = new Uri(_consulOption.Address)))
    {var services = consulClient.Catalog.Service("service-a").Result.Response;if (services != null && services.Any()){// 模拟随机一台进行请求,这里只是测试,可以选择合适的负载均衡框架var service = services.ElementAt(new Random().Next(services.Count()));var client = _httpClientFactory.CreateClient();var response = await client.GetAsync($"http://{service.ServiceAddress}:{service.ServicePort}/test/get");var result = await response.Content.ReadAsStringAsync();return result;}
    }
    
  7. 多次调用结果如下:

参考资料

[1]

Consul: https://www.consul.io/

[2]

源码下载: https://github.com/beckjin/ConsulDotnetSamples

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

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

相关文章

c语言删除最小的j个元素,最小-最大堆的插入和删除

一、定义最小-最大堆:各层交替为最小层和最大层的堆。最大层:该层上的节点大于等于以其为根节点的子树上的所有节点。最小层:该层上的节点小于等于以其为根节点的子树上的所有节点。本文中,我们令堆的层数从1开始,节点…

.NET Core + Spring Cloud:API 网关

API 网关是系统的唯一入口,调用任何服务的请求都需要经过网关层,最终才可能到达目标服务,既然是必经之路,那我们可以在网关层进行一些通用的操作,如:认证、鉴权、限流、智能路由、缓存、日志、监控、超时、…

单片机C语言中空语句,单片机C语言中的空语句.doc

头文件如://#include //包含库函数............//............_nop_(); //引用库函数敬礼。我一直都是借助仿真软件编。一点一点试时间。C语言最大的缺点就是实时性差,我在网上到看了一些关于延时的讨论,其中有篇文章51单片机 Keil C 延时程序的简单研究,…

VS Code 黑宝书背后的故事

自开售以来,《Visual Studio Code 权威指南》就受到了许多读者朋友的青睐。在京东和当当两大平台上,都分别取得了不错的绩:当当:计算机新书热卖榜第一名京东:科技IT新书榜第一名那么,热销背后,这…

c语言迪思卡尔算法,【论文】数字化的意匠_数字化设计与造型的认知学反思_谭峥.pdf...

D IG ITA L B UILD ING 数字建筑数字化的意匠—数字化设计与造型的认知学反思谭峥摘要/ 通过论述数字化设计的构成要件与理论界的思考, 数字化的设计、造型与工艺教育在西 基础。重呈一般是指对建成环境的物理状力图对数字化建构文化进行设计认知学角度的反思&#…

ASP.net Core MVC项目给js文件添加版本号

需求&#xff1a;使用ASP.net Core Mvc开发公司内部web系统&#xff0c;给视图中js(css,image也可以)文件添加版本号避免缓存问题。解决方法&#xff1a;利用Taghelper提供的标签&#xff08;asp-append-version&#xff09;可以实现<script src"~/Scripts/Biz/Village…

c语言网格搜索,基于C

引言教室作为学生长期使用的建筑类型&#xff0c;对光环境舒适度的需求尤为明显。相关研究表明&#xff0c;不仅照明会影响学习效率[1]&#xff0c;而且不当照明会引起使用者不适甚至损害视力[2]。随着多媒体教学设施的普及&#xff0c;幻灯片投影教学现已成为教师授课的主要形…

初识ABP vNext(1):开篇计划基础知识

点击上方蓝字"小黑在哪里"关注我吧审计(Audit)本地化(Localization)事件总线(Event Bus)多租户(multi-tenancy technology)DDD分层实体(Entity)值对象(Value Object)聚合根(Aggregate Root)仓储(Repository)应用服务(Application Services)数据传输对象(DTO)工作单元…

android 那几种动画,Android 动画实现几种方案

Android 动画实现几种方案在 Android 的 FrameWork 中&#xff0c;为我们提供三种动画的实现方式&#xff1a;逐帧(Frame)动画、视图/补间动画(View Animation)和属性动画(Property Animation)。由于&#xff0c;这三种动画的实现方式和针对面不一样&#xff0c;应用的范围也有…

[开源] .Net ORM FreeSql 1.8.0-preview 最新动态播报

FreeSql 是 .NET 开源生态下的 ORM 轮子&#xff0c;在一些人眼里属于重复造轮子&#xff1a;不看也罢。就像昨天有位朋友截图某培训直播发给我看&#xff0c;内容为&#xff1a;“FreeSQL&#xff08;个人产品&#xff09;&#xff0c;自己玩可以&#xff0c;不要商用。ORM框架…

android socket 服务端,Android socket 服务端

Android socket 服务端Android想在Android 上跑一个 socket 服务端。把在java工程里运行起来的代码直接放到android项目里来&#xff0c;开启线程&#xff0c;创建ServerSocket对象&#xff0c;创建对象的时候报错了。W/System.err( 3998): java.net.SocketException: socket f…

懂程序员的产品经理是什么样子?

这里是Z哥的个人公众号每周五11&#xff1a;45 按时送达当然了&#xff0c;也会时不时加个餐&#xff5e;我的第「155」篇原创敬上大家好&#xff0c;我是Z哥。在互联网行业&#xff0c;产品经理和程序员之间的关系很微妙。表面看上去水火不容&#xff0c;在一方的眼里看另外一…

Jdbc连接mysql的五种连接方式

一&#xff1a;五种连接方式 直接上码 package com.wyjedu.jdbc;import com.mysql.jdbc.Driver;import java.io.FileInputStream; import java.io.FileNotFoundException; import java.io.IOException; import java.sql.Connection; import java.sql.DriverManager; import ja…

转 android anr 分析示例,[摘]Android ANR日志分析指南之实例解析

前文《[摘]Android ANR日志分析指南》也摘抄了如何分析&#xff0c;接下来通过实例解析。一、主线程被其他线程lock&#xff0c;导致死锁waiting on <0x1cd570> (a android.os.MessageQueue)DALVIK THREADS:"main" prio5 tid3 TIMED_WAIT| group"main&quo…

455. 分发饼干001(贪心算法+详解)

一&#xff1a;题目 假设你是一位很棒的家长&#xff0c;想要给你的孩子们一些小饼干。但是&#xff0c;每个孩子最多只能给一块饼干。 对每个孩子 i&#xff0c;都有一个胃口值 g[i]&#xff0c;这是能让孩子们满足胃口的饼干的最小尺寸&#xff1b;并且每块饼干 j&#xff…

android system window,Android之属性fitsSystemWindows

页面有输入框时都要考虑会不会被键盘挡住&#xff0c;于是fitsSystemWindows属性就派上了用场属性说明fitsSystemWindows属性可以让view根据系统窗口来调整自己的布局&#xff1b;简单点说就是我们在设置应用布局时是否考虑系统窗口布局&#xff0c;这里系统窗口包括系统状态栏…

初识ABP vNext(2):ABP启动模板

点击上方蓝字"小黑在哪里"关注我吧AbpHelper模块安装前言上一篇介绍了ABP的一些基础知识&#xff0c;本篇继续介绍ABP的启动模板。使用ABP CLI命令就可以得到这个启动模板&#xff0c;其中包含了一些基础功能模块&#xff0c;你可以基于这个模板来快速开发。开始首先…

linux多行变一列,多行转为一列

--创建表ifexists (select*from sysobjects where idOBJECT_ID([TempTable_Base]) and OBJECTPROPERTY(id,IsUserTable) 1)DROP TABLE[TempTable_Base]CREATE TABLE [TempTable_Base] ([id] [int] IDENTITY (1,1)NOTNULL,[guid] [varchar] (50)NULL,[code] [varchar] (50)N…

关于Dapper实现读写分离的个人思考

概念相关为了确保多线上环境数据库的稳定性和可用性&#xff0c;大部分情况下都使用了双机热备的技术。一般是一个主库一个从库或者多个从库的结构&#xff0c;从库的数据来自于主库的同步。在此基础上我们可以通过数据库反向代理工具或者使用程序的方式实现读写分离&#xff0…

135. 分发糖果002(贪心算法+思路+详解)

一&#xff1a;题目 老师想给孩子们分发糖果&#xff0c;有 N 个孩子站成了一条直线&#xff0c;老师会根据每个孩子的表现&#xff0c;预先给他们评分。 你需要按照以下要求&#xff0c;帮助老师给这些孩子分发糖果&#xff1a; 每个孩子至少分配到 1 个糖果。 评分更高的孩…