使用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中使用的服务名称如果要用连字符,都要用短横线连接的规范形式。

相关推荐
hy.z_7775 分钟前
【数据结构】反射、枚举 和 lambda表达式
android·java·数据结构
從南走到北14 分钟前
JAVA青企码协会模式系统源码支持微信公众号+微信小程序+H5+APP
java·微信·微信小程序·小程序·uni-app·微信公众平台
草履虫建模28 分钟前
Ajax原理、用法与经典代码实例
java·前端·javascript·ajax·intellij-idea
强哥叨逼叨33 分钟前
别被假象迷惑!揭秘 Java 线程池中“线程空着但任务卡着”的真相
java
_extraordinary_36 分钟前
Java 栈和队列
java·开发语言
codervibe1 小时前
无微信依赖!纯网页扫码登录实现方案详解
java·后端
间彧1 小时前
RedisTemplate介绍与使用
java·redis
icecreamstorm1 小时前
JDBC数据库连接池
java·mysql
毕设源码柳学姐1 小时前
计算机毕业设计Java医学生在线学习平台系统 基于 Java 的医学生在线学习平台设计与开发 Java 医学在线教育学习系统的设计与实现
java·学习·课程设计
程序无bug1 小时前
5年Java开发经验,面试挂在MySQL InnoDB上!大厂究竟多看重MySQL?
java·后端