使用Spring Cloud LoadBalancer报错java.lang.IllegalStateException

1 问题描述

在使用Spring Cloud LoadBalancer进行负载均衡时,遇到错误:

而我的代码是这样写的:

java 复制代码
String url = "http://product_service/product/" + orderInfo.getProductId();
ProductInfo productInfo = restTemplate.getForObject(url, ProductInfo.class);

这里使用的restTemplate是使用@LoadBalanced注解的,因此会对该url进行解析,但是解析报错。

2 问题解决

我首先考虑了服务的名称是否正确:

其次输出服务实例的名称,发现输出的名称是大写格式,就把URL服务名称修改为大写,仍然不能解决问题。

然后我怀疑是Spring Cloud内部可能会做字符格式替换之类的,进行了如下几组实验,目前的三种情况如下:

|-----------------|-----------------|------|
| yml文件服务名称 | URL服务名称 | 实验结果 |
| product_service | product-service | 报错 |
| product-service | product-service | 不报错 |
| product_service | product_service | 报错 |
| product-service | product_service | 报错 |

URL中为短横线格式时,就不会报错。而第1组实验报错是因为实例名称不匹配,对于第3组实验报错,因此推测可能是LoadBalancer不能解析下划线:

查阅资料发现,LoadBalancer解析时强制不能使用下划线,需要使用短横线进行单词连接。这是由于LoadBalancer的源码中核心是LoadBalancerInterceptor类,它会把所有的请求都拦截,并进行解析,在该类的intercept方法中使用getHost()获取URL中的主机名(也就是服务实例名称):

java 复制代码
    public ClientHttpResponse intercept(final HttpRequest request, final byte[] body, final ClientHttpRequestExecution execution) throws IOException {
        URI originalUri = request.getURI();
        String serviceName = originalUri.getHost();
        Assert.state(serviceName != null, "Request URI does not contain a valid hostname: " + originalUri);
        return (ClientHttpResponse)this.loadBalancer.execute(serviceName, this.requestFactory.createRequest(request, body, execution));
    }

在getHost()方法内部已经标注了主机名的命名规范:

即只接受-作为连字符来连接标签,如果不符合规范就返回为null。因此使用product_service作为URL中服务实例名称就无法解析。

修改后,问题得到解决。建议URL和yml中使用的服务名称如果要用连字符,都要用短横线连接的规范形式。

相关推荐
whltaoin几秒前
SpringCloud 项目阶段九:Kafka 接入实战指南 —— 从基础概念、安装配置到 Spring Boot 实战及高可用设计
spring boot·spring cloud·kafka
callJJ26 分钟前
从 0 开始理解 Spring 的核心思想 —— IoC 和 DI(2)
java·开发语言·后端·spring·ioc·di
wangjialelele29 分钟前
Linux中的线程
java·linux·jvm·c++
谷咕咕31 分钟前
windows下python3,LLaMA-Factory部署以及微调大模型,ollama运行对话,开放api,java,springboot项目调用
java·windows·语言模型·llama
没有bug.的程序员1 小时前
MVCC(多版本并发控制):InnoDB 高并发的核心技术
java·大数据·数据库·mysql·mvcc
在下村刘湘1 小时前
maven pom文件中<dependencyManagement><dependencies><dependency> 三者的区别
java·maven
不务专业的程序员--阿飞2 小时前
JVM无法分配内存
java·jvm·spring boot
李昊哲小课2 小时前
Maven 完整教程
java·maven
Lin_Aries_04213 小时前
容器化简单的 Java 应用程序
java·linux·运维·开发语言·docker·容器·rpc
脑花儿3 小时前
ABAP SMW0下载Excel模板并填充&&剪切板方式粘贴
java·前端·数据库