SpringCloud Gateway 动态路由配置导致cpu满载,内存耗完

SpringCloud Gateway导致cpu满载,内存耗完

问题发现

**
程序打好包丢上测试环境之后,正常运行了一段时间,可是一个周末回来,发现服务器卡得不行,一开始以为是机器问题,指定堆栈大小重启然后内存再给大一点,结果过了几天还是一样,于是开始怀疑程序问题了。
**

问题查找

ps aux|head -1;ps aux|grep -v PID|sort -rn -k +3|head

查询服务器的占用情况,发现springcloud gateway程序cpu占用率99%,内存300+%。

问题分析

这个明显是一个内存溢出的问题,查询了一下issue和搜索一下相关问题,并没有相似的情况,只能自己着手分析了。

// 查询最高占用的进程id
top -Hp 6666
//看下线程的内容
jstack 7777

查询一下线程的的信息,发现并没有占用特别高的线程,用jstack看了下也没有发现奇怪的地方,但是还是得重新看下。看着看着一个 Listener 的单词引起了我的注意,想起了代码中的某一个部分。

我的网关代码分两类,一类是filter,一类是动态路由配置。动态路由配置使用到了监听器。

public class NacosRouteDefinitionRepository implements RouteDefinitionLocator {
    @Override
    public Flux<RouteDefinition> getRouteDefinitions() {
        try {
            String content = NacosFactory.createConfigService(nacosConfigProperties.assembleConfigServiceProperties()).getConfig(DATA_ID, GROUP_ID,5000);
            List<RouteDefinition> routeDefinitions = getListByYAML(content);
            return Flux.fromIterable(routeDefinitions);
        } catch (NacosException e) {
            log.error("getRouteDefinitions by nacos error", e);
        }
        return Flux.fromIterable(new ArrayList());
    }

    /**
     * 添加Nacos监听
     */
    private void addListener() {
        try {
            NacosFactory.createConfigService(nacosConfigProperties.assembleConfigServiceProperties()).addListener(DATA_ID, GROUP_ID, new Listener() {
                @Override
                public Executor getExecutor() {
                    return null;
                }

                @Override
                public void receiveConfigInfo(String configInfo) {
                    publisher.publishEvent(new RefreshRoutesEvent(this));
                }
            });
        } catch (NacosException e) {
            log.error("nacos-addListener-error", e);
        }
    }
  }

我立马禁用了动态路由配置重新启动,然后继续分析。
经过一轮的debug之后,发现 getRouteDefinitions() 这个方法是每隔几秒钟就会执行一次,这时候终于发现问题所在了。

String content = NacosFactory.createConfigService(nacosConfigProperties.assembleConfigServiceProperties())
					.getConfig(DATA_ID, GROUP_ID,5000);

这个 NacosFactory.createConfigService() 是会创建一个ConfigService对象的,这段代码相当于每隔几秒钟就创建了一个对象,这个对象gc比较难回收,所以就产生内存爆了,内存爆了cpu也跟着爆了。

动态配置代码

最后给个动态路由配置代码,也是参考了各路文章。
配置文件: application.yml

#动态更新路由开关
 gateway:
   dynamicRoute:
     enabled: true
     dataId: dynamic-routes
     groupId: TEST_GROUP

配置类: DynamicRouteConfig.class

@Configuration
@ConditionalOnProperty(prefix = "gateway.dynamicRoute", name = "enabled", havingValue = "true")
public class DynamicRouteConfig {

    @Autowired
    private ApplicationEventPublisher publisher;

    /**
     * Nacos实现方式
     */
    @Configuration
    public class NacosDynRoute {

        @Value("${gateway.dynamicRoute.dataId}")
        private String dataId;
        @Value("${gateway.dynamicRoute.groupId}")
        private String groupId ;
        @Autowired
        private NacosConfigManager nacosConfigManager;

        @Bean
        public NacosRouteDefinitionRepository nacosRouteDefinitionRepository() {
            return new NacosRouteDefinitionRepository(publisher, nacosConfigManager,dataId,groupId);
        }
    }
}

NacosRouteDefinitionRepository.class

@Slf4j
public class NacosRouteDefinitionRepository implements RouteDefinitionLocator {

    private String dataId ;
    private String groupId ;

    /**
     * RouteDefinitionLocator里面的getRouteDefinitions方法是每隔几秒钟就会执行一次的。
     * 配置修改的情况比较少,等监听到有修改的时候再做更新。
     */
    private static boolean IS_UPDATE = false;
    private List<RouteDefinition> routeDefinitions;

    private ApplicationEventPublisher publisher;

    private NacosConfigManager nacosConfigManager;

    public NacosRouteDefinitionRepository(ApplicationEventPublisher publisher, NacosConfigManager nacosConfigManager, String dataId, String groupId) {
        this.dataId = dataId;
        this.groupId = groupId;
        this.publisher = publisher;
        this.nacosConfigManager = nacosConfigManager;
        addListener();
    }

    @Override
    public Flux<RouteDefinition> getRouteDefinitions() {
        try {
            if (routeDefinitions == null || IS_UPDATE){
                String content = nacosConfigManager.getConfigService().getConfig(dataId, groupId,5000);
                routeDefinitions = getListByYAML(content);
                IS_UPDATE = false;
            }
        } catch (NacosException e) {
            log.error("getRouteDefinitions by nacos error", e);
        }
        return Flux.fromIterable(routeDefinitions);
    }

    /**
     * 添加Nacos监听
     */
    private void addListener() {
        try {
            nacosConfigManager.getConfigService().addListener(dataId, groupId, new Listener() {
                @Override
                public Executor getExecutor() {
                    return null;
                }

                @Override
                public void receiveConfigInfo(String configInfo) {
                    IS_UPDATE = true;
                    publisher.publishEvent(new RefreshRoutesEvent(this));
                }
            });
        } catch (NacosException e) {
            log.error("nacos-addListener-error", e);
        }
    }

    /**
     * json方式
     * @param content
     * @return
     */
    private List<RouteDefinition> getListByStr(String content) {
        if (StringUtils.isNotEmpty(content)) {
            return JSONObject.parseArray(content, RouteDefinition.class);
        }
        return new ArrayList<>(0);
    }

    /**
     * yml方式
     * @param content
     * @return
     */
    private List<RouteDefinition> getListByYAML(String content) {
        if (StringUtils.isNotEmpty(content)) {
            return Arrays.asList(new Yaml().loadAs(content, Route.class).getRoutes());
        }
        return new ArrayList<>(0);
    }
}
  • 1
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: Spring Cloud Gateway动态路由是指在Spring Cloud Gateway网关中,路由规则可以在运行时动态地进行添加、删除、修改等操作。相比于静态路由,动态路由可以根据实际情况进行动态调整,从而更加灵活、方便地进行流量控制和负载均衡。例如,在服务上线、下线或者进行扩容缩容的时候,可以通过修改路由规则,动态地将流量引导到不同的服务实例中,从而实现动态负载均衡和容错能力。 ### 回答2: Spring Cloud Gateway动态路由是指在Spring Cloud Gateway网关中,根据某些条件动态地将请求路由到不同的目标服务实例上。传统的静态路由需要事先配置好路由规则,但是在微服务架构中,服务的实例会动态地增加、减少、更新,因此需要一种能够动态适应变化的路由机制。 Spring Cloud Gateway动态路由的实现需要依赖于服务注册与发现组件,比如Eureka或Consul。当服务实例注册到服务注册中心时,Spring Cloud Gateway会订阅服务注册中心的变化,当有新的服务实例上线或下线时会自动更新路由规则。 动态路由可以根据多种条件进行判断和匹配,如路径、域名、Header、请求参数等。可以根据业务需求动态地配置路由规则,使得请求能够被准确地路由到目标服务实例上。动态路由能够实现动态扩展和负载均衡,提高系统的灵活性和可伸缩性。 Spring Cloud Gateway动态路由配置通常以YAML或JSON的形式进行,可以通过配置文件、配置中心或接口的方式进行配置。支持多种动态路由配置方式,如断言(Predicate)、过滤器(Filter)、转发(Forwarding)、重定向(Redirecting)等,可以根据具体需求实现各种功能。 总之,Spring Cloud Gateway动态路由是一种能够根据条件动态路由请求到不同服务实例的机制,具有灵活、可扩展、高效的特点,是构建微服务架构中的网关的重要特性。 ### 回答3: Spring Cloud Gateway动态路由是一种基于Spring Cloud Gateway框架的动态路由功能。传统的静态路由是在网关的配置文件中预先定义好所有的路由规则,而动态路由可以在运行时根据业务需要实时插入、修改和删除路由规则,实现灵活的请求转发和负载均衡。通过动态路由,可以根据不同的路径或者请求头等匹配条件,将请求转发到指定的目标服务,从而实现微服务架构中的请求路由和负载均衡功能。动态路由配置可以通过网关的API接口或者命令行工具进行管理,使得路由的配置更加灵活和方便。同时,动态路由还支持动态修改和重载路由规则,可以根据实际情况动态调整路由策略,提高系统的可用性和弹性。Spring Cloud Gateway动态路由的实现是基于Spring Framework中的路由器和过滤器的概念,通过使用reactive编程模型处理请求,并且支持使用各种插件来扩展网关的功能,例如服务发现、熔断器、限流等。总之,Spring Cloud Gateway动态路由提供了一种灵活、易用且高性能的路由解决方案,适用于构建微服务架构的API网关。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值