作者引言

很高兴啊,我们来到了IceRPC之深入理解调度管道->快乐的RPC,为上篇的续篇,深入理解常见的调度类型, 基础引导,有点小压力,打好基础,才能让自已不在迷茫,快乐的畅游世界。

传入请求

了解如何处理传入的请求

接收传入的请求

调度器的调度方法接受传入的请求。该传入请求是由连接,在收到来自对等点的请求时创建的。

请求持有如下内容:

  • 目标服务的路径
  • 服务上的操作名称
  • 请求字段
  • 请求的有效负载payload

传入请求还包含功能features。这些功能用于该调度管道内的本地通信;它们还用于管道中的调度与应用程序代码之间的通信。

请求有效负载 payload

传入请求的有效负载是表示操作参数的字节流。IceRPC 而言,该流中的字节数是未知的。

请求功能

调度管道中的调度员在调度期间相互传输信息是很常见的。C# 中,这些调度获取并设置请求的 IFeatureCollection 用于这些通信。

还可以使用这些功能与服务代码进行通信。例如,如果安装调度信息中间件,它会设置 IDispatchInformationFeature,并且可以在代码中检索此功能:

// In Slice service implementation
public ValueTask OpAsync(string message, FeatureCollection features, CancellationToken cancellationToken)
{
if (features.Get<IDispatchInformationFeature> is IDispatchInformationFeature dispatchInformation)
{
EndPoint from = dispatchInformation.ConnectionContext.TransportConnectionInformation.RemoteNetworkAddress;
Console.WriteLine($"dispatching request from {from}");
}
Console.WriteLine(message);
return default;
}

按照惯例,这些功能是使用接口类型进行键控的,例如上面示例中的 IDispatchInformationFeature

字段用于"传输连接"进行通信,而功能用于调度管道内的本地通信。IceRPC同时提供请求字段(由请求承载)和响应字段(由响应承载),但只提供请求特性:由于它都是本地的,因此不需要响应特性。

传出响应Outgoing response

了解如何创建传出响应。

创建传出响应

调度程序异步返回传出响应。

传出响应携带如下内容:

  • 状态代码
  • 错误消息,仅在状态代码非OK时设置
  • 响应字段
  • 响应的有效负载payload

响应有效负载

响应的有效负载,是表示操作返回值的字节流。调用者(发送传入请求的连接)读取这些字节,并逻辑地复制到网络连接,直到不再有字节需要读取。

C#中,传出响应的有效负载,被分割为有效负载和有效负载延续,就像传出请求的有效负载一样。这种分割,使得响应有效负载的编码,对于Slice生成的代码来说,更加方便和高效,但在其他方面,是不必要的。出发响应有效负载,在概念上是一个连续的字节流。

中间件Middleware

了解如何安装和编写中间件。

拦截传入的请求

中间件是在传入请求到达目标服务之前拦截传入请求的代码。相同的代码还会在服务发送回呼叫者之前拦截服务提供的传出响应。

在技术层面上,中间件是调度程序,它保存另一个调度程序("下一个")并在下一个调度程序上调用调度,作为其自己的调度方法实现的一部分。下一个调度程序可以是另一个中间件、服务、路由器或其他类型的调度程序;就中间件而言,它只是另一个调度程序。

中间件可以包括在下一个调度程序调用调度之前(在处理请求之前)和在下一个调度程序调用调度之后(在收到响应之后)的逻辑。 中间件还可以通过返回缓存响应或返回错误(具有 Ok 以外的状态代码的响应)来短路调度管道。

例如,一个简单的 C# 中间件可能如下所示:

public class SimpleMiddleware : IDispatcher
{
private readonly IDispatcher _next; public SimpleMiddleware(IDispatcher next) => _next = next; public async ValueTask<OutgoingResponse> DispatchAsync(
IncomingRequest request,
CancellationToken cancellationToken)
{
Console.WriteLine("before _next.DispatchAsync");
OutgoingResponse response = await _next.DispatchAsync(request, cancellationToken);
Console.WriteLine($"after _next.DispatchAsync; the response status code is {response.StatusCode}");
return response;
}
}

安装中间件

可以使用路由器,创建调度管道并在此调度管道中安装一个或多个中间件。

路由 Router

了解如何根据路径路由传入的请求。

基于路径的路由

路由是根据请求的路径,将传入的请求,路由到其他调度器的调度器。它还可以沿着这条路线执行中间件。

---
title: Path-based request routing
---
flowchart LR
subgraph Router
direction LR
m1["middleware #1"] --> m2["middleware #2"] -- /greeter --> s1[" greeter service mapped at /greeter"]
m2 -- /user/joe --> s2["account service mounted at /user"]
m2 -- /user/bob --> s2
end
connection --> m1

这些其他调度程序使用mapmount方法在路由器上注册。

  • map 将调度程序与路由器中的路径相关联。

    例如,可以将路径/greeter映射到chatbot服务,这是一场完全匹配的。具有路径/greeter2/greeter/foo 的请求是不匹配。

    C#中,会像如下代码一样:

    var router = new Router();
    router.Map("/greeter", chatbot);
  • mount 将调度程序与路由器中的路径前缀相关联。

    例如,您可以将路径前缀/user挂载到account服务。具有路径/user

    /user/foo 的请求是匹配的。而具有路径 /, /user2的请求是不匹配。

    C#中, 代码如下:

    var router = new Router();
    router.Mount("/user", account);

映射子叶调度程序(例如服务和安装子路由器)很常见,但这并不是一个硬性规则。

以映射并安装完全相同的路径(例如/greeter)。路由器将带有路径/greeter的请求,引导到映射的调度器,并将带有路径/greeter/foo的请求,引导到安装的调度器。

如果路由器没有找到,传入请求路径的映射或安装调度程序,则它会返回具有状态代码 NotFound 的响应。

子路由

子路由器是在另一个"父"路由器注册的路由器。它有一个与其安装点相对应的前缀;当它查找通过mapmount注册的调度员时,它会删除该前缀。

---
title: Path-based request routing with a sub-router
---
flowchart LR
subgraph Router
direction LR
m1["middleware #1"] --> m2["middleware #2"] -- /greeter --> s1["service #1"]
end
subgraph Sub-router
m3["middleware #3"] -- /superAdmin --> s2["service #2"]
end
m2 -- /admin/superAdmin --> m3
connection --> m1

C# 中, 可以创建一个子路由, 并使用 Route 扩展方法单步安装:

var router = new Router();

// create a sub-router and mount it at /admin
router.Route("/admin", subRouter => subRouter.UseDispatchInformation().Map("/superAdmin", root));

此示例的根服务的完整路径是/admin/superAdmin。管理子路由器在尝试将此路径与其映射和挂载字典中的条目匹配之前,会从请求的路径中删除/admin

在路由中安装中间件

路由可以在将请求,移交给映射或安装的调度程序之前执行一个或多个中间件。

在 C# 中,这些中间件通过类路由上的 Use{Name} 扩展方法注册。例如:

Router router = new Router().UseLogger(loggerFactory).UseCompressor();
router.Map("/greeter", new Chatbot());

安装这些中间件的顺序通常很重要。安装的第一个中间件,是第一个要执行的中间件。通过上面的示例,记录器中间件首先执行,然后在压缩器中间件上调用 DispatchAsync,最后压缩器中间件在/greeter 映射的 Chatbot 服务上调用 DispatchAsync

路由总是将传入的请求发送到其注册的中间件,即使它最终返回带有状态代码 NotFound 的响应。

收尾

基础概念难啊,不好写,怕写的不好,误导大家,大家看看,不用太深入,以官方为主。

作者结语

  • 一直做,不停做,才能提升速度
  • 翻译的不好,请手下留情,谢谢
  • 如果对我有点小兴趣,如可加我哦,一起探讨人生,探讨道的世界
  • 觉得还不错的话,点个

IceRPC之深入理解调度管道->快乐的RPC的更多相关文章

  1. .Net Core MVC理解新管道处理模型、中间件

    .Net Core中间件官网:https://docs.microsoft.com/zh-cn/aspnet/core/fundamentals/middleware/?view=aspnetcore ...

  2. Redis管道理解

    Redis管道理解 简介 管道并不是Redis本身提供的功能,通常是客户端提供的功能: 管道就是打包多条无关命令批量执行,以减少多个命令分别执行消耗的网络交互时间(TCP网络交互),可以显著提升Red ...

  3. Linux学习笔记(13)-进程通信|命名管道

    匿名管道只能在具有亲属关系的进程间通信,那么如果想要在不具有亲戚关系,想在陌生人之间通信,那又该怎么办呢? 别慌,Linux身为世界上*强大的操作系统,当然提供了这种机制,那便是命名管道-- 所谓命名 ...

  4. linux 有名管道(FIFO)

    http://blog.csdn.net/firefoxbug/article/details/8137762 linux 有名管道(FIFO) 管道的缓冲区是有限的(管道制存在于内存中,在管道创建时 ...

  5. go语言调度器源代码情景分析之一:开篇语

    专题简介 本专题以精心设计的情景为线索,结合go语言最新1.12版源代码深入细致的分析了goroutine调度器实现原理. 适宜读者 go语言开发人员 对线程调度器工作原理感兴趣的工程师 对计算机底层 ...

  6. Python之路(第三十九篇)管道、进程间数据共享Manager

    一.管道 概念 管道可用于具有亲缘关系进程间的通信,有名管道克服了管道没有名字的限制,因此,除具有管道所具有的功能外,它还允许无亲缘关系进程间的通信. 先画一幅图帮助大家理解下管道的基本原理 现有2个 ...

  7. Linux内核入门到放弃-进程管理和调度-《深入Linux内核架构》笔记

    进程优先级 硬实时进程 软实时进程 普通进程 O(1)调度.完全公平调度器 抢占式多任务处理(preemptive multitasking):各个进程都分配到一定的时间段可以执行.时间段到期后,内核 ...

  8. Kubernetes容器调度

    Kubernetes的调度器是Kubernetes众多组件的一部分,独立于API服务器之外.调度器本身是可插拔的,任何理解调度器和API服务器之间调用关系的工程师都可以编写定制的调度器.本文后面的介绍 ...

  9. 管道和FIFO 一

    管道和FIFO   管道(pipe)       管道在Unix及Linux进程间通信是最基础的,很容易理解.管道就像一个自来水管,一端注入水,一端放出水,水只能在一个方向上流动,而不能双向流动.管道 ...

  10. Java中的管道流 PipedOutputStream和PipedInputStream

    我们在学习IO流的时候可能会学字节流.字符流等,但是关于管道流的相信大部分视频或者教程都是一语带过,第一个是因为这个东西在实际开发中用的也不是很多,但是学习无止境,存在既有理.JDK中既然有个类那说明 ...

随机推荐

  1. 【版本发布公告】HMS Core6.5.0来啦

    新 能 力 3D Engine 3D Engine提供高性能.高画质.高扩展性的实时3D引擎,并提供便捷高效的可视化开发工具.开发者可基于华为的3D Studio开发工具,通过图形和渲染.动画.UI等 ...

  2. 财务人提高竞争力必备的技能,怎么能少了ta!

    从近年来大数据技术的发展趋势和相关产业飞速发展的状态,可以看出当前"数据分析"的热度可以说是有增无减,而且从市场上对数据分析人才的需求缺口也可以看出企业对数据分析的重视程度.未来随 ...

  3. innoSetup打包文件编写模板

    现在打包主要是使用 innosetup 这个软件来进行打包,支持录制脚本和手动编写脚本,比较好用. 此文章主要记录手写脚本,便于后期查询,借鉴. 文档: inno setup :https://blo ...

  4. 高并发报错too many clients already或无法创建线程

    高并发报错 too many clients already 或无法创建线程 本文出处:https://www.modb.pro/db/432236 问题现象 高并发执行 SQL,报错"so ...

  5. 最全能的AI换脸软件,FaceFusion下载介绍(可直播)

    FaceFusion是一款多功能的AI换脸软件,它不仅能图片.视频换脸,还可以直播换脸,换脸效果真实.自然 与大多数换脸软件不同的是,FaceFusion不仅支持N卡处理程序(Azure),还额外提供 ...

  6. Leetcode-队列得最大值

    请定义一个队列并实现函数 max_value 得到队列里的最大值,要求函数max_value.push_back 和 pop_front 的均摊时间复杂度都是O(1).若队列为空,pop_front ...

  7. 力扣344(java & python)-反转字符串(简单)

    题目: 编写一个函数,其作用是将输入的字符串反转过来.输入字符串以字符数组 s 的形式给出. 不要给另外的数组分配额外的空间,你必须原地修改输入数组.使用 O(1) 的额外空间解决这一问题. 示例 1 ...

  8. 亿图version 9.2安装教程

    记录一下自己安装亿图9.2版本的安装过程~ 先获取安装资料: 百度网盘链接: 链接:https://pan.baidu.com/s/1zJDcF9Y0Xy2CvD4mG_oOfQ?pwd=pqy9 提 ...

  9. 【阿里云采购季】3月采购完,IT运维躺赢一年

    阿里云2020上云采购季正式上线啦!今年的采购季可以逛些啥? 采购季正式期时间: 3月2日-3月31日 在这段时间里,想买啥就买吧,别忘了把想买的产品加入购物车噢,特惠产品叠加购物车满减,更划算噢! ...

  10. Sentinel 1.7.2 发布,完善开源生态及扩展性

    多样化的适配模块 到目前为止,Sentinel 已覆盖微服务.API Gateway 和 Service Mesh 三大板块的核心生态,同时多语言已推出 Java.C++.Go 三种语言的原生实现. ...