zuul是spring cloud提供的一个通用api网关组件,主要提供了服务理由、过滤器和负载均衡等功能,同时它也提供了比hystrix fallback factory粒度更粗的服务降级机制,下面就具体来看一下zuul的使用和原理。
使用zuul其实很简单,只需要一个@EnableZuulProxy注解和一个application.yml配置文件就足够了,就行下面这样
对于zuul filter的使用其实也很简单,生命一个实现了ZuulFilter的类,然后将其注册到spring 容器中就可了
另外,由于zuul处于调用链的最前端,所以一般也会做限流方面的工作,说到限流,主流的方案一般有两种,一种是漏桶算法,一种是令牌桶算法,漏桶算法能够保证任何时候进入系统内部的请求的速率是一致的,但是无法处理过多的瞬时请求,而令牌桶算法是以固定的速率生成令牌,直到到达上限,漏桶算法能够响应瞬时的大流量,知道令牌被消耗完,具体使用哪种策略还要看具体的场景,另外有一个基于zuul实现的限流工具包 zuul-ratelimit,可以实现多维限流。
下面来看一下zuul的实现原理,zuul本质是就是一个HttpServlet,下面从代码的角度来看一下。
按照惯例,先来看一下@EnableZuulProxy这个注解
到这里可以看到又是一个Marker,也就是spring cloud style,按照惯例,再去找spring.factories
由于ZuulProxyAutoConfiguration继承 了ZuulServerAutoConfiguration,所以我们直接看ZuulProxyAutoConfiguration就好了
可以看到熟悉的Marker Condition,在父类中初始化了一个Servlet
进一步的我们可以再看看这个Servlet的实现
这三个代码块分别就对应的ZuulFilter的三个生命周期方法,为了验证猜想,可以到preRoute()这个方法中看一看,深入到最后,我们会看到这样一个方法
到此,ZuulFilter的原理也就搞清楚了,实际上就是HttpServlet的service方法的一个判断分支。Zuul的另外一个主要功能是服务的路由,服务路由其实是由一个内置的Filter完成的,我们来看一下源码实现,其实也很简单
看到上面两个红框的位置,相信大家都已经清楚了,首先从根据请求的url从我们制定的配置中找到的服务名称,然后将ip和端口替换成服务名称,但这里只是名称上的替换,并没有继续真正的路由转发,这一步其实是由另外一个Filter完成的RibbonRoutingFilter,来看一下源码实现
也就是说,zuul的请求路由功能是依赖于ribbon完成的,这一点我们从zuul的pom依赖中也能够看到
这样就与eureka注册中心联系起来了。