.NET Core微服务一:Consul服务中心
.NET Core微服务二:Ocelot API网关
.NET Core微服务三:polly熔断与降级
本文的项目代码,在文章结尾处可以下载。
本文使用的环境:Windows10 64位 + VS 2019 + .NET Core 3.1
Consul有关介绍:https://www.jianshu.com/p/32dc52f28a14
Consul官方网址:https://www.consul.io/
一、安装Consul
1.官网下载对应版本,并解压出来
2.打开cmd,cd到解压的目录,运行如下其中一条命令
本次演示都在同一台计算机上,所以本文选择命令一;这两条命令请在开发环境使用,生产环境不要使用。
命令一:“consul.exe agent -dev”服务中心的ip地址,将会使用127.0.0.1
命令二:“consul.exe agent -dev -client x.x.x.x”指定服务中心的ip地址
3.命令运行成功后,打开“http://127.0.0.1:8500/”,出现如下画面
二、创建服务
1.服务端项目大体结构:
2.新建.NET Core类库项目,命名为“ClassLibrary”,然后NuGet搜索并安装
“Consul”、“Microsoft.Extensions.Configuration”
Common.cs代码:
1 using Consul;
2 using Microsoft.Extensions.Configuration;=
3 using System;
4
5 namespace ClassLibrary
6 {
7 public class Common
8 {
9 public static IConfiguration Configuration { get; }
10 static Common()
11 {
12 Configuration = new ConfigurationBuilder().AddJsonFile("appsettings.json", false, true).Build();
13 }
14
15 ///
16 /// 需要注册的服务地址
17 ///
18 public static string ConsulServiceIP
19 {
20 get
21 {
22 return Configuration["ConsulService:IP"];
23 }
24 }
25
26 ///
27 /// 需要注册的服务端口
28 ///
29 public static int ConsulServicePort
30 {
31 get
32 {
33 string str = Configuration["ConsulService:Port"];
34 return int.Parse(str);
35 }
36 }
37
38 ///
39 /// 服务注册
40 ///
41 public static void ConsulRegister()
42 {
43 ConsulClient client = new ConsulClient(
44 (ConsulClientConfiguration c) =>
45 {
46 c.Address = new Uri(Configuration["ConsulCenter:Address"]); //Consul服务中心地址
47 c.Datacenter = Configuration["ConsulCenter:DataCenter"]; //指定数据中心,如果未提供,则默认为代理的数据中心。
48 }
49 );
50 string checkUrl = Configuration["ConsulCenter:CheckUrl"];
51 client.Agent.ServiceRegister(new AgentServiceRegistration()
52 {
53 ID = Guid.NewGuid().ToString(), //服务编号,不可重复
54 Name = Configuration["ConsulService:Name"], //服务名称
55 Port = ConsulServicePort, //本程序的端口号
56 Address = ConsulServiceIP, //本程序的IP地址
57 Check = new AgentServiceCheck()
58 {
59 DeregisterCriticalServiceAfter = TimeSpan.FromMilliseconds(1), //服务停止后多久注销
60 Interval = TimeSpan.FromSeconds(5), //服务健康检查间隔
61 Timeout = TimeSpan.FromSeconds(5), //检查超时的时间
62 HTTP = $"http://{ConsulServiceIP}:{ConsulServicePort}{checkUrl}" //检查的地址
63 }
64 });
65 }
66
67 }
68 }
3.新建.NET Core的webapi项目,命名为“ServiceStudent”,把“为HTTPS配置”的勾选去掉
Controller代码:
1 using Microsoft.AspNetCore.Mvc;
2 using System;
3 using System.Collections.Generic;
4
5 namespace ServiceStudent.Controllers
6 {
7 [Route("api/[controller]/[action]")]
8 [ApiController]
9 public class DefaultController : ControllerBase
10 {
11 static List list = new List() {
12 new Student(){ ID = "001", StudentName = "学生1", StudentAge = 16 },
13 new Student(){ ID = "002", StudentName = "学生2", StudentAge = 18 },
14 new Student(){ ID = "003", StudentName = "学生3", StudentAge = 17 }
15 };
16
17 ///
18 /// 健康检查接口
19 ///
20 [HttpGet]
21 public string Check()
22 {
23 return "1";
24 }
25
26 [HttpGet]
27 public List GetList()
28 {
29 Console.WriteLine(DateTime.Now.ToString());
30 return list;
31 }
32
33 [HttpGet]
34 public Student GetModel(string id)
35 {
36 Console.WriteLine(DateTime.Now.ToString());
37 return list.Find(t => t.ID == id);
38 }
39 }
40 }
其中有一段健康检查的接口,作用是让服务中心知道服务没有挂掉还能访问,不需要什么业务,所以怎么简单怎么来:
Model代码:
public class Student
{
public string ID { get; set; }
public string StudentName { get; set; }
public int StudentAge { get; set; }
}
appsettings.json加入:
"ConsulCenter": {
"Address": "http://127.0.0.1:8500",
"CheckUrl": "/api/Default/Check",
"DataCenter": "dc1"
},
"ConsulService": {
"IP": "127.0.0.1",
"Port": 33331,
"Name": "Student"
}
Program.cs的CreateHostBuilder方法改为:
public static IHostBuilder CreateHostBuilder(string[] args)
{
string ip = Common.ConsulServiceIP;
int port = Common.ConsulServicePort;
return Host.CreateDefaultBuilder(args)
.ConfigureWebHostDefaults(webBuilder =>
{
webBuilder.UseStartup().UseUrls($"http://{ip}:{port}");
});
}
在Startup.cs的Configure方法中加入“ClassLibrary.Common.ConsulRegister();”
这样一个“Student”服务就创建好了,“Teacher”服务也是如法炮制,但是需要改appsettings.json的服务名和端口号,代码就不贴了,请到文章结尾处下载。
三、启动服务
不能在VS中启动,因为我们已经在Program.cs自定义了IP和PORT。
打开cmd,cd到Debug目录,各自运行“dotnet ServiceStudent.dll”和“dotnet ServiceTeacher.dll”。
请检查端口有没有被占用或超出范围,如果失败了,在配置文件中换成端口试试。
服务运行成功后,打开“http://127.0.0.1:8500/”,如下画面注册成功
如果出现一项打X,那就是失败,这个服务会在1分钟之后被踢出列表
一个服务可能会部署到多个服务器上,用“Teacher”来模拟下多台机器运行。
前面运行好的服务不要关闭,打开Teacher的Debug目录下的配置文件,修改它的端口号,服务名不用改,再新开一个cmd,运行“dotnet ServiceTeacher.dll”
那么服务中心就会有变化
四、使用服务
1.客户端项目大体结构:
2.新建.net core类库项目,命名为“ClassLibrary”,然后NuGet搜索并安装
“Consul”、“Microsoft.Extensions.Configuration”
Common.cs代码:
using Consul;
using Microsoft.Extensions.Configuration;
using System;
using System.Linq;
using System.Net.Http;
using System.Threading.Tasks;
namespace ClassLibrary
{
public class Common
{
public static IConfiguration Configuration { get; }
static Common()
{
Configuration = new ConfigurationBuilder().AddJsonFile("appsettings.json", false, true).Build();
}
public static string ConsulAddress
{
get { return Configuration["ConsulAddress"]; }
}
///
/// 获取服务
///
public static string GetService(string serviceName)
{
ConsulClient client = new ConsulClient(c => c.Address = new Uri(ConsulAddress));
var response = client.Agent.Services().Result.Response;
//服务名称区分大小写,若要不区分:Equals(serviceName, StringComparison.OrdinalIgnoreCase)
var services = response.Where(s => s.Value.Service.Equals(serviceName)).Select(s => s.Value);
//进行取模,随机取得一个服务器,或者使用其它负载均衡策略
var service = services.ElementAt(Environment.TickCount % services.Count());
return service.Address + ":" + service.Port;
}
///
/// 获取服务(异步方法)
///
public async Task<string> GetService2(string serviceName)
{
ConsulClient client = new ConsulClient(c => c.Address = new Uri(ConsulAddress));
var response = (await client.Agent.Services()).Response;
//服务名称区分大小写,若要不区分:Equals(serviceName, StringComparison.OrdinalIgnoreCase)
var services = response.Where(s => s.Value.Service.Equals(serviceName)).Select(s => s.Value);
//进行取模,随机取得一个服务器,或者使用其它负载均衡策略
var service = services.ElementAt(Environment.TickCount % services.Count());
return service.Address + ":" + service.Port;
}
public static string HttpGetString(string url)
{
HttpClient httpClient = new HttpClient();
string result = httpClient.GetAsync(url)
.Result.Content.ReadAsStringAsync().Result;
httpClient.Dispose();
return result;
}
public static T HttpGetObject(string url)
{
string result = HttpGetString(url);
return Newtonsoft.Json.JsonConvert.DeserializeObject(result);
}
}
}
3.新建.NET Core的webapi项目,命名为“ClientSite”,把“为HTTPS配置”的勾选去掉
StudentController代码:
1 using ClassLibrary;
2 using Microsoft.AspNetCore.Mvc;
3 using System.Collections.Generic;
4
5 namespace ClientSite.Controllers
6 {
7 [Route("api/[controller]/[action]")]
8 [ApiController]
9 public class StudentController : ControllerBase
10 {
11 [HttpGet]
12 public object GetList()
13 {
14 string ip = Common.GetService("Student");
15 List list = Common.HttpGetObject>($"http://{ip}/api/Default/GetList");
16 return new
17 {
18 address = ip,
19 data = list
20 };
21 }
22
23 [HttpGet]
24 public object GetModel(string id)
25 {
26 string ip = Common.GetService("Student");
27 Student model = Common.HttpGetObject($"http://{ip}/api/Default/GetModel?id=" + id);
28 return new
29 {
30 address = ip,
31 data = model
32 };
33 }
34 }
35 }
TeacherController代码:
1 using ClassLibrary;
2 using Microsoft.AspNetCore.Mvc;
3 using System.Collections.Generic;
4
5 namespace ClientSite.Controllers
6 {
7 [Route("api/[controller]/[action]")]
8 [ApiController]
9 public class TeacherController : ControllerBase
10 {
11 [HttpGet]
12 public object GetList()
13 {
14 string ip = Common.GetService("Teacher");
15 List list = Common.HttpGetObject>($"http://{ip}/api/Default/GetList");
16 return new
17 {
18 address = ip,
19 data = list
20 };
21 }
22
23 [HttpGet]
24 public object GetModel(string id)
25 {
26 string ip = Common.GetService("Teacher");
27 Teacher model = Common.HttpGetObject($"http://{ip}/api/Default/GetModel?id=" + id);
28 return new
29 {
30 address = ip,
31 data = model
32 };
33 }
34 }
35 }
appsettings.json加入:
"ConsulAddress": "http://127.0.0.1:8500"
4.用VS启动站点,然后用postman访问
“http://localhost:3336/api/Student/GetList”
“http://localhost:3336/api/Student/GetModel?id=003”
多次访问“http://localhost:3336/api/Teacher/GetList”,则address会随机切换,注意看返回的端口号
代码:https://files.cnblogs.com/files/shousiji/netcore_wfw.rar