011 SpringCloud 学习笔记7-----Zuul网关
1.Zuul网关概述
通过前面的学习,使用Spring Cloud实现微服务的架构基本成型,大致是这样的:
我们使用Spring Cloud Netflix中的Eureka实现了服务注册中心以及服务注册与发现;而服务间通过Ribbon或Feign实现服务的消费以及均衡负载。为了使得服务集群更为健壮,使用Hystrix的融断机制来避免在微服务架构中个别服务出现异常时引起的故障蔓延。
2.入门案例
(1)
(2)修改配置文件application.yml
server:
port: 10010 #服务端口
spring:
application:
name: service-zuul #注册到eureka后的微服务的名称
(3)修改引导类
package lucky.zuul; import org.springframework.boot.SpringApplication; import org.springframework.boot.autoconfigure.SpringBootApplication; import org.springframework.cloud.netflix.zuul.EnableZuulProxy; @SpringBootApplication @EnableZuulProxy //启用zuul组件 public class LuckyZuulApplication { public static void main(String[] args) { SpringApplication.run(LuckyZuulApplication.class, args); } }
(4)
server:
port: 10010 #服务端口
spring:
application:
name: service-zuul #注册到eureka后的微服务的名称
zuul:
routes:
service-provider: # 这里是路由id,可以随意写,但习惯上写服务名称
path: /service-provider/** # 这里是映射路径
url: http://localhost:8081 # 映射路径对应的实际url地址
3.
<dependency> <groupId>org.springframework.cloud</groupId> <artifactId>spring-cloud-starter-netflix-eureka-client</artifactId> </dependency>
(2)在lucky-zuul模块的application.xml文件中
server:
port: 10010 #服务端口
spring:
application:
name: service-zuul #注册到eureka后的微服务的名称
zuul:
routes:
service-provider: #这里是路由id,可以随意写,但习惯上写服务名称
path: /service-provider/** # 映射路径
url: http://localhost:8081 # 映射路径对应的实际url地址
eureka:
client:
registry-fetch-interval-seconds: 5 # 获取服务列表的周期:5s
service-url:
defaultZone: http://localhost:10086/eureka
(3)
package lucky.zuul; import org.springframework.boot.SpringApplication; import org.springframework.boot.autoconfigure.SpringBootApplication; import org.springframework.cloud.client.discovery.EnableDiscoveryClient; import org.springframework.cloud.netflix.zuul.EnableZuulProxy; @SpringBootApplication @EnableZuulProxy //启用zuul组件 @EnableDiscoveryClient public class LuckyZuulApplication { public static void main(String[] args) { SpringApplication.run(LuckyZuulApplication.class, args); } }
注意:以上3步是将lucky-zuul模块注册eureka注册中心中。
(4)
server:
port: 10010 #服务端口
spring:
application:
name: service-zuul #注册到eureka后的微服务的名称
zuul:
routes:
service-provider: #这里是路由id,可以随意写,但习惯上写服务名称
path: /service-provider/** # 映射路径
serviceId: service-provider # 映射路径对应的服务名
eureka:
client:
registry-fetch-interval-seconds: 5 # 设置获取服务列表的周期为5s
service-url:
defaultZone: http://localhost:10086/eureka
(5)测试
再次启动,这次Zuul进行代理时,会利用Ribbon进行负载均衡访问:
4.过滤器
Zuul作为网关的其中一个重要功能,就是实现请求的鉴权。而这个动作我们往往是通过Zuul提供的过滤器来实现的。
(1)ZuulFilter
ZuulFilter是过滤器的顶级父类。在这里我们看一下其中定义的2个最重要的方法:
package com.netflix.zuul; import com.netflix.zuul.exception.ZuulException; public interface IZuulFilter { boolean shouldFilter(); Object run() throws ZuulException; }
shouldFilter:返回一个Boolean值,判断该过滤器是否需要执行。返回true执行,返回false不执行。
run:过滤器的具体业务逻辑。
(2)过滤器执行生命周期
这张是Zuul官网提供的请求生命周期图,清晰的表现了一个请求在各个过滤器的执行顺序。
正常流程:
- 请求到达首先会经过pre类型过滤器,而后到达route类型,进行路由,请求就到达真正的服务提供者,执行请求,返回结果后,会到达post过滤器。而后返回响应。
异常流程:
- 整个过程中,pre或者route过滤器出现异常,都会直接进入error过滤器,在error处理完毕后,会将请求交给POST过滤器,最后返回给用户。
- 如果是error过滤器自己出现异常,最终也会进入POST过滤器,将最终结果返回给请求客户端。
- 如果是POST过滤器出现异常,会跳转到error过滤器,但是与pre和route不同的是,请求不会再到达POST过滤器了。
(3)使用场景
场景非常多:
- 请求鉴权:一般放在pre类型,如果发现没有访问权限,直接就拦截了
- 异常处理:一般会在error类型和post类型过滤器中结合来处理。
- 服务调用时长统计:pre和post结合使用。
(4)自定义过滤器
接下来我们来自定义一个过滤器,模拟一个登录的校验。基本逻辑:如果请求中有access-token参数,则认为请求有效,放行。
package lucky.zuul.filter; import com.netflix.zuul.ZuulFilter; import com.netflix.zuul.context.RequestContext; import com.netflix.zuul.exception.ZuulException; import io.micrometer.core.instrument.util.StringUtils; import org.apache.http.HttpStatus; import org.springframework.stereotype.Component; import javax.servlet.http.HttpServletRequest; @Component //将该类放入spring容器 public class LoginFilter extends ZuulFilter { /** * 过滤器的类型: pre、route、post、error * @return */ @Override public String filterType() { return "pre"; } /**执行顺序:返回值越小,优先级越高 * @return */ @Override public int filterOrder() { return 10; } /** * 是否执行run方法 * @return */ @Override public boolean shouldFilter() { return true; } /** * 编写过滤器的业务逻辑 * @return * @throws ZuulException */ @Override public Object run() throws ZuulException { // 获取zuul提供的上下文对象 RequestContext context = RequestContext.getCurrentContext(); // 从上下文对象中获取请求对象 HttpServletRequest request = context.getRequest(); // 获取token信息 String token = request.getParameter("access-token"); // 判断 if (StringUtils.isBlank(token)) { // 过滤该请求,不对其进行路由 context.setSendZuulResponse(false); // 设置响应状态码,401 context.setResponseStatusCode(HttpStatus.SC_UNAUTHORIZED); // 设置响应信息 context.setResponseBody("request error"); } // 校验通过,把登陆信息放入上下文信息,继续向后执行 context.set("token", token); return null; } }
(5)测试
没有token参数时,访问失败:
添加token参数后:
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 记一次.NET内存居高不下排查解决与启示
· 探究高空视频全景AR技术的实现原理
· 理解Rust引用及其生命周期标识(上)
· 浏览器原生「磁吸」效果!Anchor Positioning 锚点定位神器解析
· 没有源码,如何修改代码逻辑?
· 分享4款.NET开源、免费、实用的商城系统
· 全程不用写代码,我用AI程序员写了一个飞机大战
· MongoDB 8.0这个新功能碉堡了,比商业数据库还牛
· 白话解读 Dapr 1.15:你的「微服务管家」又秀新绝活了
· 上周热点回顾(2.24-3.2)