学风建设网站的优势,青岛正规网站设计公司,开办时 网站建设费 科目,门户网站建设的公司前言
不知道各位在使用 SpringCloud Gateway Nacos的时候有没有遇到过服务刚上线偶尔会出现一段时间的503 Service Unavailable#xff0c;或者服务下线后#xff0c;下线服务仍然被调用的问题。而以上问题都是由于Ribbon或者LoadBalancer的默认处理策略有关#xff0c;其…前言
不知道各位在使用 SpringCloud Gateway Nacos的时候有没有遇到过服务刚上线偶尔会出现一段时间的503 Service Unavailable或者服务下线后下线服务仍然被调用的问题。而以上问题都是由于Ribbon或者LoadBalancer的默认处理策略有关其中Ribbon默认是 30s 更新一次服务信息LoadBalancer则是默认 35s 更新一次缓存。接下来本文讲解则如何通过监听Nacos 的服务变更事件来实时进行相关服务的更新以实现服务的平滑上下线。
监听 Nacos 服务变更实现
首先我们要知道的是在前言中提到的服务上线未被及时感知是由于使用Ribbon或者LoadBalancer组件的默认处理策略所导致的Nacos是可以及时感知并触发服务上下线的事件因为我们要做的就是监听Nacos的这个事件然后在事件处理中自己手动去调用相关的更新操作以实现需求。而com.alibaba.nacos.client.naming.event.InstancesChangeEvent这个事件则正是符合我们需求的事件然后我们就可以参考com.alibaba.nacos.client.naming.event.InstancesChangeNotifier这个类来实现一个我们自己的订阅类
import com.alibaba.nacos.api.common.Constants;
import com.alibaba.nacos.client.naming.event.InstancesChangeEvent;
import com.alibaba.nacos.common.notify.NotifyCenter;
import com.alibaba.nacos.common.notify.listener.Subscriber;
import com.alibaba.nacos.common.notify.Event;
import com.netflix.loadbalancer.*;
import lombok.extern.slf4j.Slf4j;
import org.springframework.stereotype.Component;
import javax.annotation.PostConstruct;/*** 服务变更监听** author butterfly* date 2023-09-24*/
Slf4j
Component
public class ServiceChangeNotifier extends SubscriberInstancesChangeEvent {PostConstructpublic void init() {// 注册当前自定义的订阅者以获取通知NotifyCenter.registerSubscriber(this);}Overridepublic void onEvent(InstancesChangeEvent event) {String serviceName event.getServiceName();// 使用 dubbo 时包含 rpc 服务类会注册以 providers: 或者 consumers: 开头的服务// 由于不是正式的服务, 这里需要进行排除, 如果未使用 dubbo 则不需要该处理if (serviceName.contains(:)) {return;}// serviceName 格式为 groupNamenameString split Constants.SERVICE_INFO_SPLITER;if (serviceName.contains(split)) {serviceName serviceName.substring(serviceName.indexOf(split) split.length());}log.info(服务上下线: {}, serviceName);// 针对服务进行后续更新操作}Overridepublic Class? extends Event subscribeType() {return InstancesChangeEvent.class;}}nacos-client 为 2.1.1 时会出现订阅失效的 bug需要重写以下方法 Override
public boolean scopeMatches(InstancesChangeEvent event) {return true;
}具体原因参考issue。 基于 Ribbon 的实现
Ribbon默认情况下是 30s 刷新一次服务列表详情可看com.netflix.loadbalancer.PollingServerListUpdater其中部分代码如下
public class PollingServerListUpdater implements ServerListUpdater {private static int LISTOFSERVERS_CACHE_REPEAT_INTERVAL 30 * 1000; // msecs;private static long getRefreshIntervalMs(IClientConfig clientConfig) {return clientConfig.get(CommonClientConfigKey.ServerListRefreshInterval,LISTOFSERVERS_CACHE_REPEAT_INTERVAL);}
}这里的时间间隔可以通过ribbon.ServerListRefreshIntervalxxx进行配置其中xxx对应自定义的毫秒时间间隔而通过监听Nacos的服务变更事件则不必调整时间间隔即可实现服务的平滑上下线具体代码如下
import com.alibaba.nacos.api.common.Constants;
import com.alibaba.nacos.client.naming.event.InstancesChangeEvent;
import com.alibaba.nacos.common.notify.NotifyCenter;
import com.alibaba.nacos.common.notify.listener.Subscriber;
import com.alibaba.nacos.common.notify.Event;
import com.netflix.loadbalancer.*;
import lombok.extern.slf4j.Slf4j;
import org.springframework.stereotype.Component;
import org.springframework.cloud.netflix.ribbon.SpringClientFactory;
import javax.annotation.PostConstruct;
import javax.annotation.Resource;
import java.util.Optional;/*** 服务变更监听** author butterfly* date 2023-09-24*/
Slf4j
Component
public class ServiceChangeNotifier extends SubscriberInstancesChangeEvent {Resourceprivate SpringClientFactory springClientFactory;PostConstructpublic void init() {// 注册当前自定义的订阅者以获取通知NotifyCenter.registerSubscriber(this);}Overridepublic void onEvent(InstancesChangeEvent event) {String serviceName event.getServiceName();// 使用 dubbo 时包含 rpc 服务类会注册以 providers: 或者 consumers: 开头的服务// 由于不是正式的服务, 这里需要进行排除, 如果未使用 dubbo 则不需要该处理if (serviceName.contains(:)) {return;}// serviceName 格式为 groupNamenameString split Constants.SERVICE_INFO_SPLITER;if (serviceName.contains(split)) {serviceName serviceName.substring(serviceName.indexOf(split) split.length());}log.info(服务上下线: {}, serviceName);// 手动更新服务列表// 如果自定义负载均衡方式则将默认的 ZoneAwareLoadBalancer 替换为自己的实现即可Optional.ofNullable(springClientFactory.getLoadBalancer(serviceName)).ifPresent(loadBalancer - ((ZoneAwareLoadBalancer?) loadBalancer).updateListOfServers());}Overridepublic Class? extends Event subscribeType() {return InstancesChangeEvent.class;}}基于 LoadBalancer 的实现
默认情况下LoadBalancer的缓存时间是 35s可通过spring.cloud.loadbalancer.cache.ttl35s进行设置在org.springframework.cloud.loadbalancer.cache.DefaultLoadBalancerCacheManager类中可以看到下面是部分代码
public class DefaultLoadBalancerCacheManager implements LoadBalancerCacheManager {private SetDefaultLoadBalancerCache createCaches(String[] cacheNames,LoadBalancerCacheProperties loadBalancerCacheProperties) {// loadBalancerCacheProperties.getTtl().toMillis() 则是进行缓存的设置return Arrays.stream(cacheNames).distinct().map(name - new DefaultLoadBalancerCache(name,new ConcurrentHashMapWithTimedEviction(loadBalancerCacheProperties.getCapacity(),new DelayedTaskEvictionScheduler()),loadBalancerCacheProperties.getTtl().toMillis(), false)).collect(Collectors.toSet());}}同样的通过监听Nacos的事件我们可以在服务上下线时使相应的缓存失效即可
import com.alibaba.nacos.api.common.Constants;
import com.alibaba.nacos.client.naming.event.InstancesChangeEvent;
import com.alibaba.nacos.common.notify.NotifyCenter;
import com.alibaba.nacos.common.notify.listener.Subscriber;
import com.alibaba.nacos.common.notify.Event;
import com.netflix.loadbalancer.*;
import lombok.extern.slf4j.Slf4j;
import org.springframework.stereotype.Component;
import org.springframework.cache.Cache;
import org.springframework.cache.CacheManager;
import org.springframework.cloud.loadbalancer.core.CachingServiceInstanceListSupplier;
import javax.annotation.PostConstruct;
import javax.annotation.Resource;/*** 服务变更监听** author butterfly* date 2023-09-24*/
Slf4j
Component
public class ServiceChangeNotifier extends SubscriberInstancesChangeEvent {/*** 由于会有多个类型的 CacheManager bean, 这里的 defaultLoadBalancerCacheManager 名称不可修改*/Resourceprivate CacheManager defaultLoadBalancerCacheManager;PostConstructpublic void init() {// 注册当前自定义的订阅者以获取通知NotifyCenter.registerSubscriber(this);}Overridepublic void onEvent(InstancesChangeEvent event) {String serviceName event.getServiceName();// 使用 dubbo 时包含 rpc 服务类会注册以 providers: 或者 consumers: 开头的服务// 由于不是正式的服务, 这里需要进行排除, 如果未使用 dubbo 则不需要该处理if (serviceName.contains(:)) {return;}// serviceName 格式为 groupNamenameString split Constants.SERVICE_INFO_SPLITER;if (serviceName.contains(split)) {serviceName serviceName.substring(serviceName.indexOf(split) split.length());}log.info(服务上下线: {}, serviceName);// 手动更新服务列表Cache cache defaultLoadBalancerCacheManager.getCache(CachingServiceInstanceListSupplier.SERVICE_INSTANCE_CACHE_NAME);if (cache ! null) {cache.evictIfPresent(serviceName);}}Overridepublic Class? extends Event subscribeType() {return InstancesChangeEvent.class;}}参考资料 Nacos 自定义服务变化订阅 Spring Cloud之负载均衡组件Ribbon原理分析 SpringBoot Nacos k8s 优雅停机 微服务网关实战二SCG Nacos 服务上下线无缝切换