深入Spring Boot: 怎样排查 java.lang.ArrayStoreException
java.lang.ArrayStoreException 分析
这个demo来说明怎样排查一个spring boot 1应用升级到spring boot 2时可能出现的java.lang.ArrayStoreException
。
demo地址:https://github.com/hengyunabc/spring-boot-inside/tree/master/demo-ArrayStoreException
demo里有两个模块,springboot1-starter
和springboot2-demo
。
在springboot1-starter
模块里,是一个简单的HealthIndicator
实现
public class MyHealthIndicator extends AbstractHealthIndicator {@Overrideprotected void doHealthCheck(Builder builder) throws Exception {builder.status(Status.UP);builder.withDetail("hello", "world");}
}
@Configuration
@AutoConfigureBefore(EndpointAutoConfiguration.class)
@AutoConfigureAfter(HealthIndicatorAutoConfiguration.class)
@ConditionalOnClass(value = { HealthIndicator.class })
public class MyHealthIndicatorAutoConfiguration {@Bean@ConditionalOnMissingBean(MyHealthIndicator.class)@ConditionalOnEnabledHealthIndicator("my")public MyHealthIndicator myHealthIndicator() {return new MyHealthIndicator();}
}
springboot2-demo
则是一个简单的spring boot2应用,引用了springboot1-starter
模块。
把工程导入IDE,执行springboot2-demo
里的ArrayStoreExceptionDemoApplication
,抛出的异常是
Caused by: java.lang.ArrayStoreException: sun.reflect.annotation.TypeNotPresentExceptionProxyat sun.reflect.annotation.AnnotationParser.parseClassArray(AnnotationParser.java:724) ~[na:1.8.0_112]at sun.reflect.annotation.AnnotationParser.parseArray(AnnotationParser.java:531) ~[na:1.8.0_112]at sun.reflect.annotation.AnnotationParser.parseMemberValue(AnnotationParser.java:355) ~[na:1.8.0_112]at sun.reflect.annotation.AnnotationParser.parseAnnotation2(AnnotationParser.java:286) ~[na:1.8.0_112]at sun.reflect.annotation.AnnotationParser.parseAnnotations2(AnnotationParser.java:120) ~[na:1.8.0_112]at sun.reflect.annotation.AnnotationParser.parseAnnotations(AnnotationParser.java:72) ~[na:1.8.0_112]at java.lang.Class.createAnnotationData(Class.java:3521) ~[na:1.8.0_112]at java.lang.Class.annotationData(Class.java:3510) ~[na:1.8.0_112]at java.lang.Class.createAnnotationData(Class.java:3526) ~[na:1.8.0_112]at java.lang.Class.annotationData(Class.java:3510) ~[na:1.8.0_112]at java.lang.Class.getAnnotation(Class.java:3415) ~[na:1.8.0_112]at java.lang.reflect.AnnotatedElement.isAnnotationPresent(AnnotatedElement.java:258) ~[na:1.8.0_112]at java.lang.Class.isAnnotationPresent(Class.java:3425) ~[na:1.8.0_112]at org.springframework.core.annotation.AnnotatedElementUtils.hasAnnotation(AnnotatedElementUtils.java:575) ~[spring-core-5.0.4.RELEASE.jar:5.0.4.RELEASE]at org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerMapping.isHandler(RequestMappingHandlerMapping.java:177) ~[spring-webmvc-5.0.4.RELEASE.jar:5.0.4.RELEASE]at org.springframework.web.servlet.handler.AbstractHandlerMethodMapping.initHandlerMethods(AbstractHandlerMethodMapping.java:217) ~[spring-webmvc-5.0.4.RELEASE.jar:5.0.4.RELEASE]at org.springframework.web.servlet.handler.AbstractHandlerMethodMapping.afterPropertiesSet(AbstractHandlerMethodMapping.java:188) ~[spring-webmvc-5.0.4.RELEASE.jar:5.0.4.RELEASE]at org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerMapping.afterPropertiesSet(RequestMappingHandlerMapping.java:129) ~[spring-webmvc-5.0.4.RELEASE.jar:5.0.4.RELEASE]at org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory.invokeInitMethods(AbstractAutowireCapableBeanFactory.java:1769) ~[spring-beans-5.0.4.RELEASE.jar:5.0.4.RELEASE]at org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory.initializeBean(AbstractAutowireCapableBeanFactory.java:1706) ~[spring-beans-5.0.4.RELEASE.jar:5.0.4.RELEASE]... 16 common frames omitted
使用 Java Exception Breakpoint
下面来排查这个问题。
在IDE里,新建一个断点,类型是Java Exception Breakpoint
(如果不清楚怎么添加,可以搜索对应IDE的使用文档),异常类是上面抛出来的java.lang.ArrayStoreException
。
当断点起效时,查看AnnotationUtils.findAnnotation(Class<?>, Class<A>, Set<Annotation>) line: 686
函数的参数。
可以发现
- clazz是
class com.example.springboot1starter.MyHealthIndicatorAutoConfiguration$$EnhancerBySpringCGLIB$$945c1f
- annotationType是
interface org.springframework.boot.actuate.endpoint.annotation.Endpoint
说明是尝试从MyHealthIndicatorAutoConfiguration
里查找@Endpoint
信息时出错的。
MyHealthIndicatorAutoConfiguration
上的确没有@Endpoint
,但是为什么抛出java.lang.ArrayStoreException
?
尝试以简单例子复现异常
首先尝试直接 new MyHealthIndicatorAutoConfiguration :
public static void main(String[] args) {MyHealthIndicatorAutoConfiguration cc = new MyHealthIndicatorAutoConfiguration();}
本以为会抛出异常来,但是发现执行正常。
再仔细看异常栈,可以发现是在at java.lang.Class.getDeclaredAnnotation(Class.java:3458)
抛出的异常,则再尝试下面的代码:
public static void main(String[] args) {MyHealthIndicatorAutoConfiguration.class.getDeclaredAnnotation(Endpoint.class);}
发现可以复现异常了:
Exception in thread "main" java.lang.ArrayStoreException: sun.reflect.annotation.TypeNotPresentExceptionProxyat sun.reflect.annotation.AnnotationParser.parseClassArray(AnnotationParser.java:724)at sun.reflect.annotation.AnnotationParser.parseArray(AnnotationParser.java:531)at sun.reflect.annotation.AnnotationParser.parseMemberValue(AnnotationParser.java:355)at sun.reflect.annotation.AnnotationParser.parseAnnotation2(AnnotationParser.java:286)at sun.reflect.annotation.AnnotationParser.parseAnnotations2(AnnotationParser.java:120)at sun.reflect.annotation.AnnotationParser.parseAnnotations(AnnotationParser.java:72)at java.lang.Class.createAnnotationData(Class.java:3521)at java.lang.Class.annotationData(Class.java:3510)at java.lang.Class.getDeclaredAnnotation(Class.java:3458)
为什么会是java.lang.ArrayStoreException
再仔细看异常信息:java.lang.ArrayStoreException: sun.reflect.annotation.TypeNotPresentExceptionProxy
ArrayStoreException
是一个数组越界的异常,它只有一个String信息,并没有cause
。
那么我们尝试在 sun.reflect.annotation.TypeNotPresentExceptionProxy
的构造函数里打断点。
public class TypeNotPresentExceptionProxy extends ExceptionProxy {private static final long serialVersionUID = 5565925172427947573L;String typeName;Throwable cause;public TypeNotPresentExceptionProxy(String typeName, Throwable cause) {this.typeName = typeName;this.cause = cause;}
在断点里,我们可以发现:
- typeName是
org.springframework.boot.actuate.autoconfigure.EndpointAutoConfiguration
- cause是
java.lang.ClassNotFoundException: org.springframework.boot.actuate.autoconfigure.EndpointAutoConfiguration
终于真相大白了,是找不到org.springframework.boot.actuate.autoconfigure.EndpointAutoConfiguration
这个类。
那么它是怎么变成ArrayStoreException
的呢?
仔细看下代码,可以发现AnnotationParser.parseClassValue
把异常包装成为Object
//sun.reflect.annotation.AnnotationParser.parseClassValue(ByteBuffer, ConstantPool, Class<?>)private static Object parseClassValue(ByteBuffer buf,ConstantPool constPool,Class<?> container) {int classIndex = buf.getShort() & 0xFFFF;try {try {String sig = constPool.getUTF8At(classIndex);return parseSig(sig, container);} catch (IllegalArgumentException ex) {// support obsolete early jsr175 format class filesreturn constPool.getClassAt(classIndex);}} catch (NoClassDefFoundError e) {return new TypeNotPresentExceptionProxy("[unknown]", e);}catch (TypeNotPresentException e) {return new TypeNotPresentExceptionProxy(e.typeName(), e.getCause());}}
然后在sun.reflect.annotation.AnnotationParser.parseClassArray(int, ByteBuffer, ConstantPool, Class<?>)
里尝试直接设置到数组里
// sun.reflect.annotation.AnnotationParser.parseClassArray(int, ByteBuffer, ConstantPool, Class<?>)
result[i] = parseClassValue(buf, constPool, container);
而这里数组越界了,ArrayStoreException
只有越界的Object
的类型信息,也就是上面的
java.lang.ArrayStoreException: sun.reflect.annotation.TypeNotPresentExceptionProxy
解决问题
发现是java.lang.ClassNotFoundException: org.springframework.boot.actuate.autoconfigure.EndpointAutoConfiguration
,则加上@ConditionalOnClass
的检查就可以了:
@Configuration
@AutoConfigureBefore(EndpointAutoConfiguration.class)
@AutoConfigureAfter(HealthIndicatorAutoConfiguration.class)
@ConditionalOnClass(value = {HealthIndicator.class, EndpointAutoConfiguration.class})
public class MyHealthIndicatorAutoConfiguration {
准确来说是spring boot2把一些类的package改了:
spring boot 1里类名是:
- org.springframework.boot.actuate.autoconfigure.EndpointAutoConfiguration
spring boot 2里类名是:
- org.springframework.boot.actuate.autoconfigure.endpoint.EndpointAutoConfiguration
总结
当类加载时,并不会加载它的annotation的field所引用的
Class<?>
,当调用Class.getDeclaredAnnotation(Class<A>)
里才会加载以上面的例子来说,就是
@AutoConfigureBefore(EndpointAutoConfiguration.class)
里的EndpointAutoConfiguration
并不会和MyHealthIndicatorAutoConfiguration
一起被加载。jdk内部的解析字节码的代码不合理,把
ClassNotFoundException
异常吃掉了排查问题需要一步步深入调试
公众号
欢迎关注公众号:横云断岭的专栏,专注分享Java,Spring Boot,Arthas,Dubbo。
深入Spring Boot: 怎样排查 java.lang.ArrayStoreException相关推荐
- Spring Boot 集成Kafka java.lang.String is in module java.base of loader ‘bootstrap‘;
异常信息:java.lang.String is in module java.base of loader 'bootstrap'; com.htcyaifline.common.kafka.dom ...
- springboot 解决java.lang.ArrayStoreException
idea工具使用 Java Exception Breakpoint 添加异常断点,在IDE里,新建一个断点,类型是Java Exception Breakpoint 当断点起效时,查看Annotat ...
- java异常断点数组_使用IDEA异常断点来定位java.lang.ArrayStoreException的问题
前言 最近对 base-spring-boot项目进行了升级.在将其用于应用开发中时遇到java.lang.ArrayStoreException的异常导致程序无法启动.平常开发过程中面对这种描述不够 ...
- java.lang.ArrayStoreException
java.lang.ArrayStoreException 今天想把 List<Integer> 类型的集合转为 String[] 类型的数组 发现List类型的接口 有个toArray( ...
- actuator的原理_使用Spring Boot Actuator监视Java应用程序
actuator的原理 朋友不允许朋友写用户身份验证. 厌倦了管理自己的用户? 立即尝试Okta的API和Java SDK. 数分钟之内即可在任何应用程序中对用户进行身份验证,管理和保护. 您是否曾与 ...
- Spring Boot JPA中java 8 的应用
文章目录 Optional Stream API CompletableFuture Spring Boot JPA中java 8 的应用 上篇文章中我们讲到了如何在Spring Boot中使用JPA ...
- 使用Spring Boot Actuator监视Java应用程序
朋友不允许朋友写用户身份验证. 厌倦了管理自己的用户? 立即尝试Okta的API和Java SDK. 在几分钟之内即可对任何应用程序中的用户进行身份验证,管理和保护. 您是否曾与Spring Boot ...
- 【Spring Boot】使用Spring Boot来搭建Java web项目以及开发过程
[Spring Boot]使用Spring Boot来搭建Java web项目以及开发过程 一.Spring Boot简介 Spring Boot是由Pivotal团队提供的全新框架,其设计目的是用来 ...
- 视频教程-spring cloud与spring boot实战视频教程-Java
spring cloud与spring boot实战视频教程 从事互联网开发架构11年,曾在阿里任职java架构师,擅长SOA.分布式搜索的架构设计; 精于JVM.TCP.CPU.缓存.磁盘.网络等大 ...
最新文章
- 释放联接新价值,华为提出“1+N”5G目标网,推动运营商构筑四大数字化转型的核心能力
- SQL 存储过程 解析XML
- Oracle 数据类型及存储方式
- 如何知道osg模型每个节点的名称
- php用户名相似度对比,PHP-如何快速计算出用户的相似度
- 计算机及相关设备制造业2020,在现实生活中,接近完全垄断市场类型的行业包括()。A.计算机及相关设备制造业B.稀有...
- 糖豆人维修服务器多长时间,服务器不稳定的《糖豆人》凭啥还这么火?只因做到了这三点...
- dos 退出for循环_循环?一次帮你搞定!
- WPS2012交叉引用技巧,word比wps这点强更新參考文献
- 【转】探秘Java中的String、StringBuilder以及StringBuffer
- matlab network_无所不能的 MATLAB
- hdu7116 lowbit (线段树+lowbit的性质)
- 隐藏于世的网站内页seo优化技术
- 计算机论文源码重复太多,知网查重代码重复率太高怎么办
- MySQL事务隔离及锁机制
- 实战ItemTouchHelper仿网易新闻客户端自定义栏目页面
- java 进程 线程数量_如何查询一个进程下面的线程数(进程和线程区别)
- 【游戏程序设计】二维游戏示例-回合制Demo
- java ews appointment_ews-java-api学习:新建Recurring Appointment
- AUTOSAR MCAL解析: ADC
热门文章
- 在千锋“逆战”学习第32天 io练习(2)
- 何时开始乙肝抗病毒治疗?
- FISCO-BCOS应用实战:区块链实战应用开发分享
- 基于SSM框架便利店管理系统(进销存管理系统)(java+spring+springmvc+mybatis+maven+mysql+html)
- MySQL连接查询练习
- 关系型到文档型的跨越:颠覆你对数据库数据模型的认识
- brain怎么读_brain是什么意思_brain怎么读_brain翻译_用法_发音_词组_同反义词_脑-新东方在线英语词典...
- GPRS连接阿里云物联网平台一
- 盘点全球最热十家大数据公司中国占三席
- I.Geodetic---(弗洛伊德算法(Floyd)的运用)