ASP.NET Core结合Nacos来完成配置管理和服务发现

前言

今年4月份的时候,和平台组的同事一起调研了一下Nacos,也就在那个时候写了.net core版本的非官方版的SDK。

虽然公司内部由于某些原因最后没有真正的用起来,但很多人还是挺看好的。在和镇汐大大沟通后,决定写一篇博客简单介绍一下。

下面这个图,就是本文的重点了。

640?wx_fmt=png

Nacos的简介

Nacos是一个易于构建云原生应用的动态服务发现、配置管理和服务管理平台,它提供了一组简单易用的特性集,帮助我们快速实现动态服务发现、服务配置、服务元数据及流量管理。

它有下面的关键特性

  • 服务发现和服务健康监测

  • 动态配置服务

  • 动态 DNS 服务

  • 服务及其元数据管理

  • ...

特性还是挺多的,也有挺多值的挖掘的地方。有关Nacos的更多信息可以访问下面的地址:

  • https://nacos.io/zh-cn/

  • https://github.com/nacos-group

  • https://github.com/alibaba/nacos

下面就开始正题了,第一步肯定是先把Nacos跑起来。

启动Nacos

由于是演示,所以直接用docker启动了Standalone Mysql模式的。

git clone --depth 1 https://github.com/nacos-group/nacos-docker.git
cd nacos-docker
docker-compose -f example/standalone-mysql.yaml up

运行docker-compose后,会先拉取几个镜像回来,然后就看到下面的输出,基本就是正常启动了。

640?wx_fmt=png

打开浏览器访问 http://localhost:8848/nacos 就可以看到Nacos控制台的登录界面了。

640?wx_fmt=png

初始的用户名和密码都是 nacos,登录进来之后大概是这样的。

640?wx_fmt=png

可以看到运行起来的Nacos,版本是1.1.3,还有清晰可见的几个大菜单,这些都是可以很方便我们去进行管理的。

那我们就先来看一下Nacos的配置管理吧。

配置管理

640?wx_fmt=png

在上面的特性大图中,已经很明确的告诉了我们配置管理的几个重要功能。

在配置中有几个比较重要的概念需要先了解一下。

  • tenant 租户信息,对应 Nacos 的命名空间字段。

  • dataId 配置ID。

  • group 配置分组。

先添加下面这个nuget包,然后看一下这个配置要怎么玩。

dotnet add package nacos-sdk-csharp-unofficial

还有必不可少的就是在Startup里面进行配置。

public void ConfigureServices(IServiceCollection services)
{services.AddNacos(configure =>{configure.DefaultTimeOut = 8;configure.ServerAddresses = new System.Collections.Generic.List<string> { "localhost:8848" };configure.Namespace = "";configure.ListenInterval = 1000;});services.AddMvc().SetCompatibilityVersion(CompatibilityVersion.Version_2_2);
}

这个也算是比较常见的配置了,就不多说了,还可以通过配置文件来加载配置。

这些配置里面,其实最主要的就是Nacos的地址。

先来看看最简单的获取配置信息。

SDK中提供了一个名为INacosConfigClient的Client接口,这个接口里面的所有内容都是操作配置相关的。

[Route("api/[controller]")]
[ApiController]
public class ConfigController : ControllerBase
{private readonly INacosConfigClient _configClient;public ConfigController(INacosConfigClient configClient){_configClient = configClient;}[HttpGet("")]public async Task<string> Get([FromQuery]string key){var res = await _configClient.GetConfigAsync(new GetConfigRequest{DataId = key,Group = "DEFAULT_GROUP",}) ;return string.IsNullOrWhiteSpace(res) ? "Not Found" : res;}
}

上面获取配置的这个获取配置的方法,大意就是 读取默认命名空间(public)下面的DEFAULT_GROUP这个配置分组下面的,名为key的配置Id的值。

如果我们输入的key,在Nacos上面没有,那个这个方法就会返回 Not Found给调用方,如果有,那就会返回具体的配置值。

由于我们是刚运行起行,什么都没有操作,所以肯定是没有任何配置信息的。

640?wx_fmt=png

那我们就先添加一个,看看效果如何。

同样在上面的控制器中加入下面的发布配置的方法,同样也是通过INacosConfigClient来添加配置。


[HttpGet("add")]
public async Task<string> Add([FromQuery]string key, [FromQuery]string value)
{var res = await _configClient.PublishConfigAsync(new PublishConfigRequest{DataId = key,Group = "DEFAULT_GROUP",Content = value});return res.ToString();
}

640?wx_fmt=png

这个时候我们已经添加成功了。

\回去控制台,也可以看到刚才加的配置已经出来了。

640?wx_fmt=png

再一次访问获取配置信息的接口,就已经可以拿到对应的配置内容了。

640?wx_fmt=png

下面通过控制台去修改一下配置的内容。

640?wx_fmt=png

点发布按钮的时候,会有一个比较页面,让我们对比前后修改了那些内容。

640?wx_fmt=png

这个时候我们通过INacosConfigClient去访问的话,发现是获取不到我们刚才更新的内容的。

这个是因为,从Nacos读取配置成功后,会写入配置信息到本地缓存中,后面访问的话会优先去读缓存的内容。

那么要怎么做到有人修改了配置内容后,它能实时生效呢?其实很简单,只需要添加一下对配置的监听就可以了。

这个得益于Nacos允许我们监听配置,以便实时感知配置变更。如果配置变更,则用获取配置接口获取配置的最新值,动态刷新本地缓存。

下面是一个简单的示例,这里用的是BackgroundService来处理的。

public class ListenConfigurationBgTask : BackgroundService
{private readonly ILogger _logger;private readonly INacosConfigClient _configClient;public ListenConfigurationBgTask(ILoggerFactory loggerFactory, INacosConfigClient configClient){_logger = loggerFactory.CreateLogger<ListenConfigurationBgTask>();_configClient = configClient;}protected override async Task ExecuteAsync(CancellationToken stoppingToken){await _configClient.AddListenerAsync(new AddListenerRequest{DataId = "demo1",Callbacks = new List<Action<string>>{x =>{_logger.LogInformation($" We found something changed!!! {DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss.fff")}  [{x}]");},}});}public override async Task StopAsync(CancellationToken cancellationToken){await _configClient.RemoveListenerAsync(new RemoveListenerRequest{DataId = "demo1",Callbacks = new List<Action>{() =>{_logger.LogInformation($" Removed listerner  ");},}});await base.StopAsync(cancellationToken);}
}

这里其实没有什么内容,就是在程序启动的时候添加一下监听,然后在程序退出的时候,同样也退出监听。

不要忘记在Startup中加下面的代码,这样配置的监听才会生效!

services.AddHostedService<ListenConfigurationBgTask>();

当我们添加监听之后,修改了配置文件的内容,它就可以动态的更新加载了。

640?wx_fmt=gif

同样的,控制台里面也有监听的记录,可以在监听查询里面找到。

640?wx_fmt=png

下面是具体的程序日志输出

640?wx_fmt=png

配置的每一次修改,都会有历史记录,可以从历史版本里面找到。

640?wx_fmt=png

除了能看历史的记录,还可以回滚到指定的版本,这是个很有用的功能。

在数据库中,配置信息的保存是这样的

640?wx_fmt=png

还有一个删除配置的方法,这里就不介绍了,都是差不多的用法,不过正常情况下是不应该删除配置的,除非是多余的。

关于Nacos配置管理的介绍就先到这里了,有兴趣的朋友可以继续去深究。

下面我们就来看看Nacos的服务发现。

服务发现

关于服务注册和发现,听的比较多的大概就是,consul, eureka, etcd , k8s 等等。

思路其实都差不多,在服务启动的时候,把当前服务的相关信息注册上去,然后要调用某个服务的时候,就获取这个服务下面的列表,然后选一个可用的进行访问。最后就是当服务停止的时候,我们要注销当前的服务。

目前这个SDK提供了两种形式,一种是原始的API,一种是对原始API进行了封装,可以直接注册和发现相应的下游服务。

原始的API在一个名为INacosNamingClient的Client接口中提供,这个接口里面的所有内容都是服务发现相关的。

不过在这里只介绍封装过后的使用方法,当然也可以自己根据原始的API进行封装处理。

首先要添加下面这个nuget包。

dotnet add package nacos-sdk-csharp-unofficial.AspNetCore

先起来一个服务。

先在配置文件appsettings.json中添加下面的内容

{"nacos": {"ServerAddresses": [ "localhost:8848" ],"DefaultTimeOut": 15,"Namespace": "","ListenInterval": 1000,"ServiceName": "BaseService","Weight": 10}
}

这个配置主要表达了,这个实例的服务名是 BaseService, 权重是10, Nacos的地址是 localhost:8848

然后在Startup中把当前实例注册到Nacos。

namespace BaseService
{using Microsoft.AspNetCore.Builder;using Microsoft.AspNetCore.Hosting;using Microsoft.AspNetCore.Mvc;using Microsoft.Extensions.Configuration;using Microsoft.Extensions.DependencyInjection;using Nacos.AspNetCore;public class Startup{public Startup(IConfiguration configuration){Configuration = configuration;}public IConfiguration Configuration { get; }public void ConfigureServices(IServiceCollection services){services.AddNacosAspNetCore(Configuration);services.AddMvc().SetCompatibilityVersion(CompatibilityVersion.Version_2_2);}public void Configure(IApplicationBuilder app, IHostingEnvironment env){if (env.IsDevelopment()){app.UseDeveloperExceptionPage();}app.UseMvc();app.UseNacosAspNetCore();}}
}

这里只需要简单的配置这两个地方就可以完成服务的注册功能了!!

下面就启动这个程序。

640?wx_fmt=png

可以看到在启动程序的时候,当前实例就会向Nacos发送心跳,心跳的里面包含了IP和端口等信息。

回到控制台,我们可以看到这个服务现在已经有一个实例了。

640?wx_fmt=png

再启动一个同服务名的实例,这里只对接口返回的内容做了一下调整,其他都是一样的!

这个时候点进服务的详情里面,可以看到更加具体的信息。

640?wx_fmt=png

服务现在是已经注册上来了,下面我们就再来一个服务去调用上面这个注册好的服务。

Startup中的内容都是差不多的,不同的是,如果确定服务不被内部其它应用调用的话,可以不注册到Nacos上面。

public class Startup
{public Startup(IConfiguration configuration){Configuration = configuration;}public IConfiguration Configuration { get; }public void ConfigureServices(IServiceCollection services){services.AddHttpClient();services.AddNacosAspNetCore(Configuration);services.AddMvc().SetCompatibilityVersion(CompatibilityVersion.Version_2_2);}public void Configure(IApplicationBuilder app, IHostingEnvironment env){if (env.IsDevelopment()){app.UseDeveloperExceptionPage();}app.UseMvc();}
}

然后就是发现服务了。

INacosServerManager里面提供了一个只根据服务名来获取健康的实例的地址信息。不足的地方就是忽略了命名空间和集群这些参数,会考虑在后面的版本中加上吧。

这里获取到的地址信息是随机取出来的,最简单的轮训算法。。获取到一次所有的实例地址信息后会缓存10秒钟,这10秒钟里面就会直接从缓存中的地址信息取一个。

[Route("api/[controller]")]
[ApiController]
public class ValuesController : ControllerBase
{private readonly INacosServerManager _serverManager;private readonly IHttpClientFactory _clientFactory;public ValuesController(INacosServerManager serverManager, IHttpClientFactory clientFactory){_serverManager = serverManager;_clientFactory = clientFactory;}[HttpGet]public async Task<string> GetAsync(){var result = await GetResultAsync();if (string.IsNullOrWhiteSpace(result)){result = "ERROR!!!";}return result;}private async Task<string> GetResultAsync(){var baseUrl = await _serverManager.GetServerAsync("BaseService");if (string.IsNullOrWhiteSpace(baseUrl)){return "";}var url = $"{baseUrl}/api/values";var client = _clientFactory.CreateClient();var result = await client.GetAsync(url);return await result.Content.ReadAsStringAsync();}
}

效果就来看动图了。

640?wx_fmt=gif

在两个实例的健康状态都是true的时候,会随机调用一个实例。

当把其中一个实例停掉的时候,这个实例的健康状态就会被标识为false,这个时候就不会调用到这个false的实例。

当把这个实例重新运行之后,又恢复到随机调用的情况。

Nacos的服务发现除了上面介绍的,还有系统开关,数据指标,集群信息等功能,有待去深入挖掘。

写在最后

Nacos使用起来不算复杂,算是比较容易上手的,用的公司也挺多的了。

还有个把 steeltoe 和Nacos结合起来的项目 skynet-cloud:https://github.com/magicsgxie/skynet-cloud 也可以看看。

文中的示例代码可以戳这里 NacosDemo :https://github.com/catcherwong-archive/2019/tree/master/09/NacosDemo

SDK的地址 nacos-sdk-csharp:https://github.com/catcherwong/nacos-sdk-csharp

希望感兴趣的大佬给个星星,也十分希望有大佬来一起维护这个项目,和提些建议。

因为是第一次写SDK类的东西,参考了其他平台提供.NET的SDK,然后结合Nacos的Open API写的,有可能会有不少遗漏和bug,还请各位大佬多多包涵。

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

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

相关文章

2020 ICPC 济南 F. Gcd Product

Gcd Product Cm∑i1mAgcd⁡(i,m)Bgcd⁡(k1−i,m)∑d1∣mAd1∑d2∣mBd2∑i1m([gcd⁡(id1,md1)1][d1∣i])([gcd⁡(m1−id2,md2)1][d2∣m1−i])∑d1∣mAd1∑d2∣mBd2∑k1∣md1μ(k1)∑k2∣md2μ(k2)∑i1m([d1∣i][k1∣id1])([d2∣m1−i][k2∣m1−id2])T1d1k1,T2d2k2∑T1∣m∑d1∣T…

Java 时间处理

时区、冬令时和夏令时、时间戳 时间戳 距离一个标准参照时间经过的秒数&#xff08;毫秒数&#xff09; 有两个常用参照时间&#xff1a; 1970-01-01 00:00:00 应用最广泛的时间戳参照点2001-01-01 00:00:00 常被苹果系统使用 注意&#xff1a;以上时间节点皆采用UTC的标准时…

试试这个Excel知识测验,得分超过80分算你赢

大家可能都知道&#xff0c;全世界使用Excel的用户超过了10亿。Excel的知识真所谓是博大精深&#xff0c;并且还很有趣味。我最近编写了一个Excel小工具&#xff0c;可以让大家可以在Excel里面进行各种知识小测验&#xff0c;并且与全世界的高手一比高低。这个小工具&#xff0…

SimpleDateFormat与线程安全

SimpleDateFormat不是线程安全的。 SimpleDateFormat(下面简称sdf)类内部有一个Calendar对象引用&#xff0c;它用来储存和这个sdf相关的日期信息&#xff0c;例如sdf.parse(dateStr)&#xff0c;sdf.format(date) 诸如此类的方法参数传入的日期相关String, Date等等&#xff…

几道偏序问题(数据结构)

P3157 [CQOI2011]动态逆序对 #include <bits/stdc.h>using namespace std;typedef long long ll;const int N 1e5 10;int root[N], ls[N << 8], rs[N << 8], sum[N << 8], cnt;int n, m, pos[N];inline int lowbit(int x) {return x & (-x); }v…

自学架构设计?帮你总结了 4 个方法

从编程思维到架构思维的升级&#xff0c;是工作 3、5 年的程序员遇到的第一个槛&#xff0c;特别是当你准备晋升考核时。我有个哥们&#xff0c;技术和业务都很不错&#xff0c;腾讯 T2.3 升 T3.1&#xff0c;就卡在了架构设计这部分。架构这个事儿&#xff0c;不像算法和代码&…

如何在东八区的计算机上获取美国时间

既可以用旧API&#xff08;JDK8之前&#xff09;&#xff0c;也可以使用新API。以下用旧API为例&#xff1a; 在Java语言中&#xff0c;可以通过java.util.Calendar类取得一个本地时间或者指定时区的时间实例&#xff0c;如下&#xff1a; // 取得本地时间&#xff1a; Calen…

Wannafly挑战赛24 无限手套(生成函数)

无限手套 每种宝石的生成函数为∑n≥0xn(ain2bin1)对其进行化简∑n≥0xn∑n≥0binxn∑n≥0ain2xn11−xbix(1−x)2aix(1x)(1−x)3最后答案∏i1m((ai−bi1)x2(aibi−2)x1)(1−x)3m每种宝石的生成函数为\sum_{n \geq 0} x ^ n(a_i n ^ 2 b_i n 1)\\ 对其进行化简\sum_{n \geq 0}…

.NET Core 3.0及ASP.NET Core 3.0 前瞻

前几天微软发布了 .NET Core 3.0 Preview 9 &#xff0c;这是.NET Core 3.0 最后一个预览版。[翻译] .NET Core 3.0 Preview 9 发布.NET Core 3.0 正式发布将在.NET Conf 上发布&#xff0c;.NET Conf 时间是9月23日至25日。Visual Studio 2019 16.3预览版3和Visual Studio for…

有了Unicode为啥还需要UTF-8

有了Unicode为啥还需要UTF-8 要回答这个问题&#xff0c;需要吃透“编码”的概念&#xff0c;刚好看到大神阮一峰写的文章&#xff1a;字符编码笔记&#xff1a;ASCII&#xff0c;Unicode 和 UTF-8 抄录如下&#xff0c;便于查找&#xff1a; 一、ASCII 码 我们知道&#xf…

2019ICPC西安邀请赛 E. Tree(树剖 + 线段树)

Tree 给定一棵树&#xff0c;节点有点权&#xff0c;然后有三种操作&#xff1a; 一、修改1−>s1->s1−>s的路径上的点权与ttt进行按位或。 二、修改1−>s1->s1−>s的路径上的点权与ttt进行按位与。 三、查询1−>s1->s1−>s的路径上的点权异或和…

升职却不加薪,为什么我还觉得老板说的挺有道理

前几天晚上&#xff0c;DevOps交流群里&#xff0c;有人抛出这样一个观点&#xff1a;如果有人来找我加薪&#xff0c;我一定告诉他我要给他升职&#xff0c;因为升职是免费的&#xff0c;加薪可是真的要花钱。但是我也会许诺他&#xff0c;如果他能把那个团队搞好&#xff0c;…

GBK、GB2312、GB18030

GBK、GB2312都是编码方式&#xff0c;都是双字节字符集&#xff0c;GB18030编码是一二四字节变长编码 GB&#xff1a;国标&#xff0c;国家标准 BIG5&#xff1a; 支持繁体中文 GB2312&#xff1a; 支持简体中文&#xff08;GBK的子集&#xff09; 共收录 6763 个汉字 GBK&…

2019 ICPC 南京 F. Paper Grading(字典树dfs序上树套树)

Paper Grading 题意&#xff1a;给定nnn个字符串&#xff0c;有两种操作&#xff1a; 一、给定i,ji, ji,j&#xff0c;交换第iii个跟第jjj个字符串。 二、给定 str &#xff0c;k,l,rk, l, rk,l,r&#xff0c;问你在区间[l,r][l, r][l,r]中的字符&#xff0c;与 str 至少有k…

细节之中自有天地,整洁成就卓越代码

溪源 | 长沙.NET技术社区开篇我们总是很容易就能写出满足某个特定功能的代码&#xff0c;却很难写出优雅代码。又最欣赏那些优雅的代码&#xff0c;因为优雅代码更能体现一个开发者的积累。就像写一篇散文&#xff0c;有的就像初学者不得其门而入&#xff0c;遣词造句都非常困难…

#2693. jzptab

jzptab ∑i1n∑j1mlcm(i,j)∑i1n∑j1mijgcd⁡(i,j)∑d1nd∑i1nd∑j1mdij[gcd⁡(i,j)1]∑d1nd∑k1ndk2μ(k)∑i1nkdi∑j1mkdjTkd,f(n)∑i1ni∑T1nf(nT)f(mT)(T∑k∣Tμ(k)k)设g(n)n∑d∣nμ(d)d先令g(n)g(n)ng(1)1,g(p)μ(1)μ(p)p1−p,g(pk,k≥2)1−p同时是积性函数&#xff0c…

UTF8、UTF16、UTF32区别

UTF8、UTF16、UTF32都是unicode字符集的字符编码。 UTF意思是unicode转换格式&#xff08;Unicode transform format&#xff09;&#xff0c;出现UTF8、UTF16、UTF32是出于要在内存中存储字符的目的而对unicode字符编号进行编码。 UTF8、UTF16、UTF32区别&#xff1a;&#…

URL编解码、Big Endian和Little Endian

一、Endian的起源 在各种计算机体系结构中&#xff0c;对于字节、字等的存储机制有所不同&#xff0c;因而引发了计算机通信领域中一个很重要的问题&#xff0c;即通信双方交流的信息单元&#xff08;比特、字节、字、双字等等&#xff09;应该以什么样的顺序进行传送。如果不达…

一次业务网关用ASP.NET Core 2.1重构的小结

前言对于API网关&#xff0c;业界貌似对它进行下划分&#xff0c;有下面几个分类/场景。面向Web App面向Mobile App面向Partner OpenAPI面向Partner ExternalAPI其他。。。在18年8月份的时候&#xff0c;有幸用.NET Core 2.1重构了一个对外的业务网关项目&#xff0c;这个项目的…

导数卷积 (NTT)

导数卷积 有f(x)∑i0n−1aixi求g(x)∑i0n−1f(i)(x)f(n−i−1)(x)∑i0n−1∑j0n−1aij(ij)!j!∑k0n−1ak(n−i−1)(k(n−i−1))!k!设F(n)ann!我们单独求解g(x)的第m项&#xff0c;有gm∑i0n−1∑j1d(F(ij)∗F(n−i−1d−j))(1j!∗1(d−j)!)设H(n)1n!有f(x) \sum_{i 0} ^{n - …