springCloud通过gateway+ribbon实现负载均衡
一、简单介绍下均衡负载
为什么要实现负载均衡?我知道有两个重要的作用,一是提高服务的高可用性,二是"均衡"请求压力,都很好理解,机制就是将客户端请求分发到不同地址的服务提供方,例如现在商品服务请求太多了,一个服务承受不住了,那就横向扩容,再加一个商品服务,这两个服务的功能当然是一模一样的,两个的好处是即使其中一个服务宕机停掉了或者其他的故障另一个照样可以正常提供服务,用户感觉不出来的。
二、基于gateway实现均衡负载
今天下午逛了下各种博客,看了下都是怎么实现均衡负载的,视频也看了,都大同小异,别人的做法我觉得是有很大的问题的,比如现在B服务实现集群有两个,别人的做法是通过A服务在eureka注册中心中找到B服务然后实现均衡负载,关键代码是这样的:
public class MovieController {
private static final Logger LOGGER = LoggerFactory.getLogger(MovieController.class);
@Autowired
private RestTemplate restTemplate;
@Autowired
private LoadBalancerClient loadBalancerClient;
@GetMapping("/user/{id}")
public User findById(@PathVariable Long id) {
//VIP:virtual IP
return this.restTemplate.getForObject("http://microservice-provider-user/" + id, User.class);
}
}
流程大概是这样的,
这样得通过另一个服务调用别的服务才能实现均衡负载,明显很不合理,实际工作中肯定不会是这样干的。
我一向以最实用为准,我的做法非常简单,实现均衡负载无需写任何代码,我的做法就是将上面的服务A改为gateway网关,通过网关查找eureka中的服务进行均衡负载。
注册中心截图(我的注册中心是服务器上的,没有任何关系的)
三、具体实现(贴出主要代码)
相信看到这的人肯定对springcloud有了解的,就不把所有的代码都贴出来了,不要省事,gateway网关是微服务绕不过去的,实际工作中微服务架构必然是通过网关的端口进行访问的,网关配置起来也超级简单的。
需要启动四个项目,Eureka注册中心、gateway网关、mall-product(两个,端口号或者IP地址不一致都行,本地启动的话肯定是改个端口就行了),将gateway、mall-product全都注册到eureka中。
在mall-product中建个controller,代码如下
@RestController
@RequestMapping("/ribbon")
public class OrderController {
@Value("${server.port}")
private String port;
@GetMapping("/test")
public String testRibbon(){
return "port:"+port;
}
}
贴出下主要配置文件吧,gateway网关的配置文件如下:
server:
port: 8080 #就用最熟悉的端口吧
spring:
application:
name: gateway
cloud:
gateway:
# 开启项目名映射功能,即通过服务名访问项目,不需要写明具体IP地址
discovery:
locator:
enabled: true
# 请求路径去不转为小写字母,如AA/aB转为aa/ab
lower-case-service-id: true
eureka:
instance:
prefer-ip-address: true
# 在注册中心上显示的信息
instance-id: ${spring.application.name}:${server.port}
hostname: gateway-service
client:
service-url:
defaultZone: http://47.98.149.207:8000/eureka/
eureka服务端和mall-product的配置文件就不发了,都太简单了,由于是在自己本地的电脑上测试的,所以mall-product的两个服务只是端口不一致(实际工作大概率是分布式集群的,即在不同ip地址的服务器上启动的),mall-product一个端口为8082一个为8084。
四、使用postman测试
通过网关的端口8080访问,
默认的均衡负载机制为轮询,即每个服务轮流接收请求。