问题:
- Spring Cloud Version:Hoxton.SR3
- Spring Cloud Alibaba Version:2.2.0.RELEASE
项目中consumer端使用@FeignClient
定义接口,启动报错:
Caused by: java.lang.AbstractMethodError: com.alibaba.cloud.sentinel.feign.SentinelContractHolder.parseAndValidateMetadata(Ljava/lang/Class;)Ljava/util/List;
at feign.ReflectiveFeign$ParseHandlersByName.apply(ReflectiveFeign.java:151) ~[feign-core-10.7.4.jar:na]
at feign.ReflectiveFeign.newInstance(ReflectiveFeign.java:49) ~[feign-core-10.7.4.jar:na]
分析:
Spring Cloud从Hoxton.SR1版本开始,依赖的spring-cloud-starter-openfeign
版本为2.2.2.RELEASE;
Spring Cloud Alibaba 2.2.0.RELEASE依赖的spring-cloud-openfeign-dependencies
版本为2.2.0.RELEASE;
feign的Contract
接口,
2.2.0.RELEASE:
// TODO: break this and correct spelling at some point
List<MethodMetadata> parseAndValidatateMetadata(Class<?> targetType);
2.2.2.RELEASE:
List<MethodMetadata> parseAndValidateMetadata(Class<?> targetType);
在2.2.0.RELEASE里有一行注释描述,接口方法名拼写错误,在2.2.2.RELEASE方法已修正了,即方法名发生了改变。
在spring-cloud-alibaba-sentinel
中的SentinelContractHolder
类,用到了该接口的这个方法(feign2.2.0.RELEASE版本):
@Override
public List<MethodMetadata> parseAndValidatateMetadata(Class<?> targetType) {
List<MethodMetadata> metadatas = delegate.parseAndValidatateMetadata(targetType);
metadatas.forEach(metadata -> METADATA_MAP
.put(targetType.getName() + metadata.configKey(), metadata));
return metadatas;
}
解决:
以下3中方式均可:
-
修改Spring Cloud版本,Hoxton.SR3改为Hoxton.RELEASE
-
在dependencyManagement中指定
spring-cloud-openfeign-dependencies
的版本为2.2.0.RELEASE
<dependencyManagement>
<dependencies>
<dependency>
<groupId>org.springframework.cloud</groupId>
<artifactId>spring-cloud-openfeign-dependencies</artifactId>
<version>2.2.0.RELEASE</version>
<type>pom</type>
<scope>import</scope>
</dependency>
<dependency>
<groupId>org.springframework.cloud</groupId>
<artifactId>spring-cloud-dependencies</artifactId>
<version>Hoxton.SR3</version>
<type>pom</type>
<scope>import</scope>
</dependency>
<dependency>
<groupId>com.alibaba.cloud</groupId>
<artifactId>spring-cloud-alibaba-dependencies</artifactId>
<version>2.2.0.RELEASE</version>
<type>pom</type>
<scope>import</scope>
</dependency>
</dependencies>
</dependencyManagement>
注意:spring-cloud-openfeign-dependencies
写在spring-cloud-dependencies
前面,因为dependency前面的优先生效。
- 在项目中相同包路径下新建
SentinelContractHolder
类,方法实现和spring-cloud-alibaba-sentinel
相同,修改Contract
接口为新版版本的方法名即可。
总结:
-
AbstractMethodError是ERROR的子类,不能在编译时检查出来,运行时才会触发
-
A依赖于B,运行时JVM发现B的class文件和编译时不一致会抛该异常,一般是因为B有多个版本,版本不兼容导致,比如方法签名变了。