介绍完分布式配置中心,结合前面的文章。咱们已经有了一个微服务的框架了,能够对外提供api接口服务了。但如今试想一下,在微服务框架中,每一个对外服务都是独立部署的,对外的api或者服务地址都不是不尽相同的。对于内部而言,很简单,经过注册中心自动感知便可。但咱们大部分状况下,服务都是提供给外部系统进行调用的,不可能同享一个注册中心。同时通常上内部的微服务都是在内网的,和外界是不连通的。并且,就算咱们每一个微服务对外开放,对于调用者而言,调用不一样的服务的地址或者参数也是不尽相同的,这样就会形成消费者客户端的复杂性,同时想一想,可能微服务多是不一样的技术栈实现的,有的是
http
、rpc
或者websocket
等等,也会进一步加大客户端的调用难度。因此,通常上都有会有个api网关,根据请求的url不一样,路由到不一样的服务上去,同时入口统一了,还能进行统一的身份鉴权、日志记录、分流等操做。接下来,咱们就来了解今天要讲解的路由服务:zuul
。html
简单来讲,微服务网关是微服务架构中一个不可或缺的部分。经过服务网关统一贯外系统提供REST API的过程当中,除了具有服务路由、均衡负载功能以外,它还具有了权限控制等功能。前端
在未加入网关时,通常上会在服务外网架设一个负载均衡,如nginx等。此时,微服务的组成为:java
此时,对于Open Service
而言可能须要提供权限控制等和业务无关的能力,这样自己就破坏了微服务服务单一的原则。因此,通常上在Open Service
之上,还有一层服务提供诸如通用的权限校验、参数校验等功能,此服务就是网关了。以后,对于内部微服务而言,只须要关系各自微服务提供的业务功能便可,无需去关心其余非业务相关的功能。nginx
API网关能够提供一个单独且统一的API入口用于访问内部一个或多个API。简单来讲嘛就是一个统一入口,好比如今的支付宝或者微信的相关api服务同样,都有一个统一的api地址,统一的请求参数,统一的鉴权。git
优势:github
缺点:web
如今市场上有不少的网关可供选择:spring
Netflix
开源的微服务网关,能够和Eureka
,Ribbon
,Hystrix
等组件配合使用。spring
官方基于Spring5.0
,Spring Boot2.0
,Project Reactor
等技术开发的网关,提供了一个构建在Spring Ecosystem
之上的API网关,旨在提供一种简单而有效的途径来发送API,并向他们提供交叉关注点,例如:安全性,监控/指标和弹性。目的是为了替换Spring Cloud Netfilx Zuul
的。在Spring cloud
体系中,通常上选择zuul
或者Gateway
。固然,也能够综合本身的业务复杂性,自研一套或者改在一套符合自身业务发展的api网关的,最简单作法是作个聚合api服务
,经过SpringBoot
构建对外的api接口,实现统一鉴权、参数校验、权限控制等功能,说白了就是一个rest
服务。后端
Zuul
是Netflix
开源的微服务网关,能够和Eureka
、Ribbon
、Hystrix
等组件配合使用,Spring Cloud
对Zuul
进行了整合与加强,Zuul
的主要功能是路由转发和过滤器。api
Zuul
基于JVM的路由器和服务器端负载均衡器。同时,Zuul
的规则引擎容许规则和过滤器基本上用任何JVM语言编写,内置支持Java
和Groovy
。这个功能,就能够实现动态路由的功能了。当须要添加某个新的对外服务时,通常上不停机更新是经过数据缓存配置或者使用Groovy
进行动态路由的添加的。
Zuul
的核心一系列的过滤器:
加入Zuul
网关后:
建立工程:spring-cloud-zuul
。 这里直接加入了注册中心进行服务化模式。
0.加入pom依赖。
<!-- zuul 依赖 --> <dependency> <groupId>org.springframework.cloud</groupId> <artifactId>spring-cloud-starter-netflix-zuul</artifactId> </dependency> <!-- eureka client 依赖 --> <dependency> <groupId>org.springframework.cloud</groupId> <artifactId>spring-cloud-starter-netflix-eureka-client</artifactId> </dependency>
注意:这里的Eureka
不是必须的。在没有注册中心的状况下,也是能够进行zuul使用的。
1.配置文件,配置注册中心相关信息、路由规则等。
spring.application.name=zuul-server server.port=8888 # 注册中心地址 -此为单机模式 eureka.client.service-url.defaultZone=http://127.0.0.1:1000/eureka # 启用ip配置 这样在注册中心列表中看见的是以ip+端口呈现的 eureka.instance.prefer-ip-address=true # 实例名称 最后呈现地址:ip:15678 eureka.instance.instance-id=${spring.cloud.client.ip-address}:${server.port} ## 路由规则 ## 传统路由配置:不依赖服务发现。 ## 全部以myapi开头的url路由至http://127.0.0.1:2000/下 ## 如http://127.0.0.1:8888/myapi/hello --> http://127.0.0.1:2000/hello zuul.routes.myApi.path=/myapi/** zuul.routes.myApi.url=http://127.0.0.1:2000 #forward模式 直接转发至zuul提供的rest服务 zuul.routes.myDemo.path=/myDemo/** zuul.routes.myDemo.url=forward:/demo ## 服务发现模式 # 路由地址 zuul.routes.myEureka.path=/eureka/** #为具体服务的名称 zuul.routes.myEureka.service-id=eureka-client
友情提示: 默认状况下:Zuul代理全部注册到EurekaServer的微服务,路由规则: http://ZUUL_HOST:ZUUL_PORT/微服务实例名(serverId)/**
转发至serviceId对应的微服务。
如:http://127.0.0.1:8888/eureka-client/hello?name=oKong 最后就是转发至:http://127.0.0.1:2000//hello?name=oKong
2.启动类加入@EnableZuulProxy
注解,声明一个Zuul代理。
/** * zuul 示例 * * @author oKong * */ @SpringBootApplication @EnableZuulProxy @EnableDiscoveryClient @Slf4j public class SpringCloudZuulApplication { public static void main(String[] args) throws Exception { SpringApplication.run(SpringCloudZuulApplication.class, args); log.info("spring-cloud-zuul启动!"); } }
3.编写一个控制类,测试forward
功能。
/** * zuul 内部提供对外服务示例 * @author oKong * */ @RestController @RequestMapping("/demo") public class DemoController { @GetMapping("/hello") public String hello(String name) { return "hi," + name + ",this is zuul api! "; } }
4.启动spring-cloud-eureka-server
和spring-cloud-eureka-client
服务,以后再启动spring-cloud-zuul
服务。
此时,咱们来访问zuul内部服务:http://127.0.0.1:8888/demo/hello?name=oKong
而后访问:http://127.0.0.1:8888/myDemo/hello?name=oKong 效果和上面直接方式是同样的,最后都是访问/demo/**
api地址的。
访问:http://127.0.0.1:8888/myapi/hello?name=oKong 和 http://127.0.0.1:8888/eureka/hello?name=oKong 最后效果都是同样的,都是访问http://127.0.0.1:2000/hello?name=oKong 。
直接以默认形式访问:http://127.0.0.1:8888/eureka-client/hello?name=oKong 最后效果也是同样的。
其实,从控制台输出日志也是能够一窥究竟的:
简单说明下关于路由规则的一些说明:
zuul.routes.<route>.path
和zuul.routes.<route>.url
参数对的方式来配置# 传统路由配置 zuul.routes.server-provide.path=/server-provide/** zuul.routes.server-provide.url=http://127.0.0.1:1001/
zuul.routes.<route>.path
与zuul.routes.<route>.serviceId
参数对的方式配置# 多实例 zuul.routes.server-provide.path=/user-service/** zuul.routes.server-provide.serviceId=user-service ribbon.eureka.enabled=false server-provide.ribbon.listOfServers=http://127.0.0.1:1001/,http://127.0.0.1:1001/
默认规则:http://ZUUL_HOST:ZUUL_PORT/微服务实例名(serverId)/** ,转发至serviceId对应的微服务。
自定义路由规则:经过一组zuul.routes.<route>.path
与zuul.routes.<route>.serviceId
参数对的方式配置
# 自定义路由 zuul.routes.server-provide.path=/server-api/** zuul.routes.server-provide.serviceId=server-provide
好比:
并且,要注意,这些过滤器是path
进行最佳路径匹配的,因此,通常上在一些历史系统上,咱们会在最后后面加上一个路径/**的匹配规则,以保证历史api可使用,作到最大兼容性,避免相似404的异常。
zuul.routes.legacy.path=/**
本章节主要简单介绍了关于
Zuul
的一些简单使用以及一些路由规则的简单说明。开头也说过了,Zuul
的核心是一系列的过滤器。介于篇幅问题,关于过滤器的介绍、自定义过滤器、异常处理、熔断降级等放在下一章节来说解。而相关的其余配置,你们能够去官网进行查看,或者自行搜索下,也能够查看下:org.springframework.cloud.netflix.zuul.filters.ZuulProperties
类,相关配置信息都在里面了。
目前互联网上大佬都有分享
SpringCloud
系列教程,内容可能会相似,望多多包涵了。原创不易,码字不易,还但愿你们多多支持。若文中有错误之处,还望提出,谢谢。
499452441
lqdevOps
我的博客:http://blog.lqdev.cn
源码示例:https://github.com/xie19900123/spring-cloud-learning
原文地址:http://blog.lqdev.cn/2018/10/14/SpringCloud/chapter-nine/