Zuul2核心架构

article/2025/9/22 7:56:41

Zuul2的核心架构就是就是两大体系,netty体系filter体系

1 Netty体系

Zuul2底层采用Netty的事件响应模式,要掌握zuul2就必须先要掌握Netty。

1.1 Channel、Event、EventLoop、EventLoopGroup、ChannelHandler

Channel:每一次通信就会启动一个channel,一个channel对应一个socket
Event:事件,channel的入站和出站都会触发一些列的事件,这些事件会回调ChannelHandler的逻辑。入站:连接已被激活或者连接失活;数据读取;用户事件;错误事件。出站:打开或者关闭到远程节点的连接;将数据写到或者冲刷到套接字。
EventLoop:事件循环,控制流、多线程处理以及并发处理。每个EventLoop对应一个线程。
EventLoopGroup:负责给每个创建的channel分配一个EventLoop。

解释说明:
EventLoop与Channel是多对多的,一个线程可以支持多个channel,一旦一个Channel 被分配给一个EventLoop,它将在它的整个生命周期中都使用这个EventLoop,所以我们面向channel开发时不需要担心线程安全的问题。

1.2 ChannelPipeline、ChannelHandler、ChannelHandelerContext

ChannelHandler:每个ChannelHandler 的实例都类似于一种为了响应特定事件而被执行的回调。Handler分为入站和出站两种类型。
ChannelPipeline:一组ChannelHandler的编排,入站事件从头部开始执行,出站事件从尾部开始执行。

ChannelHandlerContext:代表了ChannelHandler 和ChannelPipeline 之间的关联,是channel信息维护的手段。

关联关系:

解释说明:
ChannelPipeline、ChannelHandler是池的概念,channel和ChannelHandlerContext的生命周期是临时性的。

1.3 BootStarp和ServerBootStarp

Netty网络编程中的2个概念,面向客户端的是BootStarp,面向服务端的是ServerBootStarp,像我们统一网关的应用场景就是面向服务端。
BootStarp:连接到远程节点的客户端应用程序。
ServerBootstrap:就是比BootStarp在前面绑定一个接口多嵌套一个ServerChannel来处理套接字,后面还是交给BootStarp与客户端交互。

2 Filter体系:

Zuul2实现了一种基于filter的体系设计,通过实现ZuulFilter来定制属于自己的网关,抛开netty不讲我们来看看这套体系是如何运作的。
这套体系核心接口:ZuulFilter、ZuulMessage、FilterRunner、FilterUsageNotifier,其中每个接口都有很深层级的继承和实现。
ZuulFiter:官宣暴漏给用户自己实现的接口,位置和运行方式两个维度。位置分为:inbound、outbound、endpoint,方式分为同步和异步。其中endpoint只有同步方式。
ZuulMessage:filter体系中传递的消息信息,核心属性是SessionContext,分为Request和Response两种message。
FilterRunner:filter体系的调度中心,filter中抽象方法都有runner来调用。分为EndPointRunner和FilterChainRunner两种。
FilterUsageNotifier:filter体系的调用通知机制,目前功能比较单一,只实现了统计功能,但是从体系设计中看得出可以扩展出更多服务能力。
UML图:


核心代码在BaseZuulFilterRunner的filter方法,传入ZuulMessage和ZuulFilter完成对他们的调度,本质上还是让zuulFilter自己消费了ZuulMessage。
ZuulFilterChainRunner完成了Filter链的调用。

3 如何结合:

上面说的Netty体系和Filter体系是Zuul2中的两大独立体系,我们直接没有看到它们之间有什么关联。因为讲Netty的时候没有提到过Filter,讲Filter的时候也没有提到过Channel。那么这两个体系是怎么关联的呢?
本质是上Filter体系作为Netty体系Pipeline中的一环,或者理解为内部的一次嵌套。秘密在ZuulFilterChainHandler和BaseZuulChannelInitializer中。
ZuulFilterChainHandler通过channelRead方法和方法,在channel的读取和自定义事件回调时织入了filter体系的逻辑,而ZuulFilterChainHandler本身又通过BaseZuulChannelInitializer的addZuulHandlers方法被织入了channel的Pipeline中。

ZuulFilterChainHandler:

@Overridepublic void channelRead(ChannelHandlerContext ctx, Object msg) throws Exception {if (msg instanceof HttpRequestMessage) {zuulRequest = (HttpRequestMessage)msg;//Replace NETTY_SERVER_CHANNEL_HANDLER_CONTEXT in SessionContextfinal SessionContext zuulCtx = zuulRequest.getContext();zuulCtx.put(NETTY_SERVER_CHANNEL_HANDLER_CONTEXT, ctx);zuulCtx.put(ZUUL_FILTER_CHAIN, requestFilterChain);requestFilterChain.filter(zuulRequest);}else if ((msg instanceof HttpContent)&&(zuulRequest != null)) {requestFilterChain.filter(zuulRequest, (HttpContent) msg);}else {LOG.debug("Received unrecognized message type. " + msg.getClass().getName());ReferenceCountUtil.release(msg);}}

BaseZuulChannelInitializer:

protected void addZuulHandlers(final ChannelPipeline pipeline){pipeline.addLast("logger", nettyLogger);pipeline.addLast(new ClientRequestReceiver(sessionContextDecorator));pipeline.addLast(passportLoggingHandler);addZuulFilterChainHandler(pipeline);pipeline.addLast(new ClientResponseWriter(requestCompleteHandler, registry));}

 简洁说,netty是外圈,filter是其中一环上的内圈;netty为zuul2提供基础设施,filter在基础设施上做上层建设。

 


http://chatgpt.dhexx.cn/article/E4iQuHDo.shtml

相关文章

【Zuul2】网关Zuul控制台DashBoard

目录 一、需求背景 二、实现方案 一、源码获取 二、源码分析 三、效果展示 三、相关问题 一、需求背景 用JAVA为开发语言的流控网关主要分为以下三种: Netflix Zuul/Zuul2Spring Cloud GateWayAlibaba Sentinel 从定位上来看,Zuul2与SpringClou…

【Zuul2】Zuul2网关介绍及案例(非spring集成)

目录 一.使用缘由 二.项目介绍 1.核心内容 (1)三种过滤器 Inbound、Endpoint 、Outbound (2)配置文件application.properties (3)动态配置application.properties 2.参考文档 一.使用缘由 公司需要在springcloudgateway和zuul2间做一次较为完整的调研对比,…

time_t c语言 2038,什么是2038问题?

什么是2038问题 不知道你有没有听过2038问题?无论你是否听过,本文将带你认识什么是2038问题。 Unix时间戳 定义为从格林威治时间1970年01月01日00时00分00秒(北京时间1970年01月01日08时00分00秒)起至现在的总秒数。 而在C语言中,常用time_t来表示。举个例子: #include #in…

MySQL的时间戳2038年问题还有16年,最好在设计上的时候使用datetime就可以了,不要使用时间戳字段了,即使用了也不要用int类型进行映射,使用long类型映射即可

目录 前言1,关于MySQL时间戳的2038年BUG2,使用Docker创建MySQL 模拟下3,总结 前言 本文的原文连接是: https://blog.csdn.net/freewebsys/article/details/127455169 未经博主允许不得转载。 博主CSDN地址是:https://blog.csdn.n…

计算机为什么不用三十二进制,32位进制导致2038年问题

该楼层疑似违规已被系统折叠 隐藏此楼查看此楼 在计算机应用上,2038年问题可能会导致某些软件在2038年无法正常工作。所有使用UNIX时 间表示时间的程序都将受其影响,因为它们以自1970年1月1日经过的秒数(忽略闰秒)来表示时间。 这种时间表示法在类Unix(Unix-like)操作系统上是…

2038年问题 linux内核5.6,Linux Kernel 5.6 开发者已率先做好准备 应对 2038 年问题

新十年伊始,Linux Kernel 5.6的开发者已经准备好着手解决将在下一个十年到来的2038年问题(又称“Y2038”或“Unix Y2K”问题)。Linux 5.6也成为第一个为32位系统准备运行到2038年之后的主线内核。 2038年问题与千年虫问题类似,它可能会导致某些软件在203…

mysql 2038年问题_时间戳(UnixTimestamp)与 《2038年问题》

时间戳是从格林威治时间1970年01月01日00时00分00秒(北京时间1970年01月01日08时00分00秒)起至现在的总秒数。 现在时间戳的长度是十位(1435113975--2015/6/24 10:46:15)。 要到 2286/11/21 01:46:40 才会变成11位(10000000000),距离现在还有 271年。 不同时区获取的…

2038年问题 linux内核5.6,Linux Kernel 5.6 开发者已准备好应对 2038 年问题

2038 年问题与千年虫问题类似,它可能会导致某些软件在 2038 年 1 月 19 日 3 时 14 分 07 秒之后无法正常工作。届时,在大部分 32 位操作系统上,依据 “time_t” 标准,时间将会“绕回”且在内部被表示为一个负数,并造成…

2038年问题 linux内核5.6,Linux 5.1延续为2038年问题所做的多年准备

Linux 5.1为2038年问题在内核方面继续进行大量的工作。 多年来在Linux内核一直看到“Y2038”的工作,而这一努力远未结束。Thomas Gleixner为Linux 5.1内核提交了最新的Y2038工作,在之前的内核中做了大量基础工作之后,Linux 5.1内核引入了一组…

2038计算机系统,2038年问题

2038年问题是指在使用POSIX时间的32位计算机应用程序上,格林尼治时间2038年1月19日凌晨03:14:07(北京时间:2038年1月19日中午11:14:07)之后无法正常工作。 中文名 2038年问题 外文名 Year 2038 problem概 念 计算机bug(程序错误) 载 体 使用POSIX时…

聊一聊2038年问题

庚子年是中国传统的 60 甲子纪年法。擅长观测的古人很早就发现,每当年份执行到庚子这一年,自然灾害变多,突发事件频频,一些震动世界、影响安定的大事件也容易发生在这一年。而我们现在所处的 2020 年就是新一轮的庚子年&#xff0…

List集合去重 --指定对象属性去重

在针对特定场景下,将获取到的list<T> 集合 按照某一个特定的对象中的属性进行去重操作,以下代码会将传入的集合进行指定去重,会将指定属性重复的对象 只保留第一个,后续的重复则不会保存到去重后的集合中,当然我们也可以通过集合的差异获取出重复的对象以及后续的再将去…

使用Set集合对List集合进行去重

使用Set集合对List集合进行去重 前段时间正好遇到这样一个需求&#xff1a;我们的支付系统从对方系统得到存储明细对象的List集合&#xff0c;存储的明细对象对象的明细类简化为如下TradeDetail类&#xff0c;需求是这样的&#xff0c;我要对称List集合进行去重&#xff0c;这里…

关于两个List集合对象去重

实际项目开发中&#xff0c;很多业务场景下都会遇见集合去重。在说到两个对象去重之前&#xff0c;首先我们回顾下普通类型的list如何去重&#xff0c;这里先说两个list自带的方法&#xff0c;图画的不是太好&#xff0c;勿喷- -&#xff01; 一&#xff1a;retainAll() List&…

java list集合数据去重方式

1.概述 最近又是一轮代码review , 发现了一些实现去重的代码&#xff0c;在使用 list.contain … 我沉思&#xff0c;是不是其实很多初学者也存在这种去重使用问题&#xff1f; 所以我选择把这个事情整出来&#xff0c;分享一下。 2.contain 去重 首先是造出一个 List 模拟…

Java【List】去重的 6种方法

list集合去重 一、HashSet去重二、TreeSet去重三、LinkedHashSet去重四、迭代器去重五、Stream去重六、contains判断去重等等... 其它实现方法 一、HashSet去重 我们知道 HashSet 天生具备“去重”的特性&#xff0c;那我们只需要将 List 集合转换成 HashSet 集合就可以了&…

List 去重的 6 种方法,这种方法最完美!

在日常的业务开发中&#xff0c;偶尔会遇到需要将 List 集合中的重复数据去除掉的场景。这个时候可能有同学会问&#xff1a;为什么不直接使用 Set 或者 LinkedHashSet 呢&#xff1f;这样不就没有重复数据的问题了嘛&#xff1f; ​ 不得不说&#xff0c;能提这个问题的同学很…

List去除重复数据的五种方式

你知道的越多&#xff0c;不知道的就越多&#xff0c;业余的像一棵小草&#xff01; 你来&#xff0c;我们一起精进&#xff01;你不来&#xff0c;我和你的竞争对手一起精进&#xff01; 编辑&#xff1a;业余草 blog.csdn.net/qq_37939251/article/details/90713643 推荐&…

List元素去重的六种方式

上周的时候完成公司交付的任务&#xff0c;突然间遇到了需要把重复元素去掉的功能&#xff0c;当时我的大脑飞速运转&#xff0c;努力回想以前学习关于list的知识&#xff0c;后来&#xff0c;我发现已经忘得差不多了&#xff0c;所以我就找到了这篇文章&#xff0c;说的很详细…

子图是什么

子图和真子图 设 G <V, E>, <, >是两个图(同为无向,或同为有向图). 若 V 且 E, 则称 为 G 的子图, G 为 的母图, 记作 G 若 V 或 E, 称 为 G 的真子图. 生成子图 若 G 且 V , 则称 为 G 的生成子图 两个导出子图 设 V 且 &#xff08;空集&a…