.net core 微服务之Api网关(Api Gateway)
原文:.net core 微服务之Api网关(Api Gateway)
1、 微服务引子
首先恭喜你,进入微服务的开发世界。微服务属于架构演进中的一种阶段,其特点是根据业务模块水平划分服务种类,每个服务可以独立部署并互相隔离,并对外提供轻量的Api调用,服务具有高可用特性。
微服务应遵循的设计原则:
- 单一职责原则: 每个微服务只需要实现自己的业务逻辑
- 服务自治原则: 每个微服务都是独立的,不依赖其他模块
- 轻量级通信原则:一般采用Http + Json方式
- 接口明确原则:接口尽量做的更通用,更灵活,从而尽量避免接口参数的来回修改。
我从2017年12月开始接触微服务概念,并开始着手构建公司的微服务平台,系统架构采用 .net core webapi方式组织,随着微服务的增多,越来越需要一个统一入口管理这些微服务。
2、使用Nginx作为api网关
Nginx是由IgorSysoev为俄罗斯访问量第二的Rambler.ru站点开发的,一个高性能的HTTP和反向代理服务器。2012年,Nginx荣获年度云计算开发奖,并成长为世界第二大Web服务器。全世界流量最高的前1000名网站中,超过25%都使用Nginx来处理海量的互联网请求。
Nginx很牛掰,业界公认的首选,选择它作为api网关,可以说不用开发介入,只需要运维的同学好好规划配置即可。

网关配置规划如下:
/api/ServiceA —> ServiceA
/api/ServiceB —> ServiceB
/api/ServiceC —> ServiceC
外部统一的访问入口是Nginx,然后根据服务名称路由到不同的微服务站点。
api网关的路由难题解决了,其他熔断,灰度发布,线上测试,日志拦截等功能nginx做起来相对比较吃力,不过对于中小型平台已经够用了。
3、自创api网关(重复轮子)
半年后,换了家公司,公司微服务平台搭建采用Thrift RPC作为各个微服务的通讯协议,由于当时的无知(Socket协议已被Nginx支持),所以决定写个Api代理类,没敢叫Api网关,是因为实现的功能仅限于路由!
仅仅是路由,当然是祭出 .net core Web应用。
3.1、构建初始化
MVC框架包就别包含了,轻量的性能才能好!
Main函数初始化线程池大小
//初始化线程池最小大小 为 300
ThreadPool.GetMinThreads(out var wt, out var ct);
if (wt < 300 || ct < 300)
{
ThreadPool.SetMinThreads(Math.Max(wt, 300), Math.Max(ct, 300));
}
**注意:**线程池不是越大越好,最佳选择约等于 cpu内核的2倍,有个公式参考:

多说一句吧,思考下:
为什么Nginx只用4个线程发挥出的性能就大大超越了100个进程的Apache HTTPD?回想一下计算机科学的基础知识,答案其实是很明显的。
我要支持跨域访问:
public void ConfigureServices(IServiceCollection services)
{
services.AddCors(options =>
{
options.AddPolicy("AllowAll", p => p.AllowAnyOrigin().AllowAnyMethod().AllowAnyHeader().AllowCredentials());
});
}
配置下使用中间件
public void Configure(IApplicationBuilder app, IHostingEnvironment env)
{
app.UseCors("AllowAll");
app.UseMiddleware<ProxyMiddleware>();
// app.UseMvc();
}
3.2、构建中间件
路由转发,一个中间件即可搞定。
public class ProxyMiddleware
{
private readonly RequestDelegate m_Next;
public ProxyMiddleware(RequestDelegate next)
{
this.m_Next = next;
}
public Task Invoke(HttpContext context)
{
if (context.Request.Method?.ToUpper() == "GET")
{
var path = context.Request.Path.HasValue ? context.Request.Path.ToString().ToLower() : string.Empty;
if (string.IsNullOrEmpty(path) || path.Equals("/") || path.Equals("/api") || path.Equals("/api/"))
{
return this.SendConstRespond(context);
}
//增加一个代理网关接口,返回微服务列表
else if (path.EndsWith("/getallservices"))
{
return this.SendServiceListRespond(context);
}
return this.SendStringRespond(context);
}
if (context.Request.ContentType == null || context.Request.ContentType.IndexOf("application/json") < 0)
{
context.Response.StatusCode = 403;
context.Response.ContentType += "application/json;charset=utf-8;";
return context.Response.WriteAsync("Please set ContentType=application/json");
}
return this.SendStringRespond(context);
}
private Task SendStringRespond(HttpContext context)
{
context.Response.StatusCode = 200;
context.Response.ContentType += "application/json;charset=utf-8;";
Task task = new Task(() =>
{
if (ServerSetting.Config.QuantumConfig.RpcService.ServerType==ServerType.HttpWebApi)
{
string constResp = QuantumHttpProxy.SendHttp(context);
using (var strStream = new StreamWriter(context.Response.Body))
{
strStream.Write(constResp);
strStream.Flush();
}
}
else
{
QuantumHttpProxy.Send(context);
}
});
task.Start();
return task;
}
private Task SendConstRespond(HttpContext context)
{
context.Response.StatusCode = 200;
context.Response.ContentType += "application/json;charset=utf-8;";
Task task = new Task(() =>
{
var constResp = new
{
rid = string.Empty,
c = 200,
msg = "Access api gateway success!"
};
using (var strStream = new StreamWriter(context.Response.Body))
{
strStream.Write(JsonConvert.SerializeObject(constResp));
strStream.Flush();
}
});
task.Start();
return task;
}
private Task SendServiceListRespond(HttpContext context)
{
context.Response.StatusCode = 200;
context.Response.ContentType += "application/json;charset=utf-8;";
Task task = new Task(() =>
{
var svrs = ServerSetting.Config?.HttpProxy?.Items?.Select(x => x.Name)?.ToList() ?? new List<string>();
var constResp = new
{
rid = Guid.NewGuid().ToString("N"),
c = 200,
v = svrs,
};
using (var strStream = new StreamWriter(context.Response.Body))
{
strStream.Write(JsonConvert.SerializeObject(constResp));
strStream.Flush();
}
});
task.Start();
return task;
}
}
上面的代码里 有 QuantumHttpProxy,是我们封装的rpc调用,因此各位同学要用此类,请自行修改之。
4、结语
没有重试,熔断也没有降级,是否很low?
这就是实际实践的力量,不必耗费太多精力在一些非关键点上!
好吧,我承认我研究了 **polly**类库,重试,熔断,降级都准备好了,你就起航吧~~~
另外:造轮子只适合特定场景,小投入就可以完成的某些特定功能,比如上面的路由功能!如果你是云服务,可以重点考虑下 阿里API网关,亚马逊api网关!
引用链接
.net core 微服务之Api网关(Api Gateway)的更多相关文章
- .NET Core微服务二:Ocelot API网关
.NET Core微服务一:Consul服务中心 .NET Core微服务二:Ocelot API网关 .NET Core微服务三:polly熔断与降级 本文的项目代码,在文章结尾处可以下载. 本文使 ...
- (8)学习笔记 ) ASP.NET CORE微服务 Micro-Service ---- Ocelot网关(Api GateWay)
说到现在现有微服务的几点不足: 1) 对于在微服务体系中.和 Consul 通讯的微服务来讲,使用服务名即可访问.但是对于手 机.web 端等外部访问者仍然需要和 N 多服务器交互,需要记忆他们的服务 ...
- 基于.NET CORE微服务框架 -谈谈surging API网关
1.前言 对于最近surging更新的API 网关大家也有所关注,也收到了不少反馈提出是否能介绍下Api网关,那么我们将在此篇文章中剥析下surging的Api 网关 开源地址:https://git ...
- .NET Core微服务一:Consul服务中心
本文的项目代码,在文章结尾处可以下载. 防爬虫,本文的网址是:https://www.cnblogs.com/shousiji/p/12253295.html 本文使用的环境:Windows10 64 ...
- .NET Core微服务之基于Ocelot实现API网关服务
Tip: 此篇已加入.NET Core微服务基础系列文章索引 一.啥是API网关? API 网关一般放到微服务的最前端,并且要让API 网关变成由应用所发起的每个请求的入口.这样就可以明显的简化客户端 ...
- .NET Core微服务之基于Ocelot实现API网关服务(续)
Tip: 此篇已加入.NET Core微服务基础系列文章索引 一.负载均衡与请求缓存 1.1 负载均衡 为了验证负载均衡,这里我们配置了两个Consul Client节点,其中ClientServic ...
- .NET Core微服务之基于Steeltoe集成Zuul实现统一API网关
Tip: 此篇已加入.NET Core微服务基础系列文章索引,本篇接上一篇<基于Steeltoe使用Eureka实现服务注册与发现>,所演示的示例也是基于上一篇的基础上而扩展的. => ...
- .net core 微服务架构-docker的部署-包括网关服务(Ocelot)+认证服务(IdentityServer4)+应用服务(asp.net core web api)
本文主要介绍通过Docker来部署通过.Net Core开发的微服务架构,部署的微服务主要包括统一网关(使用Ocelot开发).统一认证(IdentityServer4).应用服务(asp.net c ...
- .NET Core 微服务—API网关(Ocelot) 教程 [二]
上篇文章(.NET Core 微服务—API网关(Ocelot) 教程 [一])介绍了Ocelot 的相关介绍. 接下来就一起来看如何使用,让它运行起来. 环境准备 为了验证Ocelot 网关效果,我 ...
随机推荐
- 【MySQL集群】——Java程序连接MySQL集群
上篇简介了怎样在Windows环境下建立配置MySQL集群,这里用一个实现注冊功能的小Demo通过jdbc的方式连接到MySQL集群中. 外部程序想要远程连接到mysql集群,还须要做的一个操作就是设 ...
- item-设置可见性
如果我们想要设置menu中item的可见行,有两种方式: 1.直接在menu的xml代码中设置 <menu> <item android:id="@+id/action_h ...
- poi完美word转html(表格、图片、样式)
直入正题,需求为页面预览word文档,用的是poi3.8,以下代码支持表格.图片,不支持分页,只支持doc,不支持docx: /** * */ import java.io.BufferedWrite ...
- 水题ing
T1: https://www.luogu.org/problemnew/show/P1724幻想乡,东风谷早苗是以高达控闻名的高中生宅巫女.某一天,早苗终于入手了最新款的钢达姆模型.作为最新的钢达姆 ...
- eclipse中的乱码问题
在新建项目或导入工程时常常遇到的问题: 1.导入工程后,Java文件中文乱码 项目右键--Properties--Resource(快捷键Alt+Enter),在Text file encoding中 ...
- [D3] Build an Area Chart with D3 v4
Similar to line charts, area charts are great for displaying temporal data. Whether you’re displayin ...
- regexp模式匹配+location页面跳转+cookie/localstorage本地存储
学习js的过程中,根据知识点编写一些code进行测试,以便检验. 这段程序使用了以下知识点: 1.regexp,对数据进行模式匹配 2.使用location对象进行页面跳转. 3.cookie/loc ...
- python 命令行:help(),'more'不是内部或外部命令,也不是可运行的程序或批处理文件
Python下使用help(dict),显示'more'不是内部或外部命令,也不是可运行的程序或批处理文件,该如何处理? 环境变量设置的问题,进入 Path 的环境变量设置界面,将;%SystemRo ...
- MongoDbHelper 帮助类(下)
对MongoDbHelper帮助类进行了一下整合,但是代码中一个方法需要将string类型转化为BsonValue类型一直出错.所以欢迎留言指正 using System; using System. ...
- Centos 6 DNS Server 配置
安装bind yum install -y bind bind-chroot bind-utis 如果是Centos 5 # yum -y install bind caching-nameserve ...