提供给其他系统使用的jar包造成tomcat启动报错:
Unable to process Jar entry [module-info.class] from Jar [jar:file:/xxxxxxxx/lombok-1.18.4.jar!/] for annotations
org.apache.tomcat.util.bcel.classfile.ClassFormatException: Invalid byte tag in constant pool: 19
问题信息搜集:
有人给lombok提了bug(https://github.com/rzwitserloot/lombok/issues/1801) tomcat8.0环境启动读取module-info.class报错
然后一个作者回复:我们加了java9的module-info是有原因的, 应该换tomcat版本或者给tomcat提bug兼容这个 大概意思是这样
进行了一些信息搜集之后得出:从lombok-1.16.20开始打包中存在一个叫module-info.class的文件,一些版本的tomcat启动时,如果classpath下有这个包就会报上面的错误
尝试的解决方案:
1.maven中将lombok的scope设为provider;——有效,lombok为注解增强编译,运行时不需要
2.对方系统tomcat版本为7.0.61,本地换用tomcat 8.0.53;——有效,未发现异常
问题相关知识梳理:
bcel - [Byte Code Engineering Library] apache旗下的字节码处理框架
tomcat中实现了bcel的功能,lib/tomcat-coyote.jar中的包com.apache.tomcat.util.bcel为相关类
抛出上面的异常【Invalid byte tag in constant pool: 19】的原因:
at org.apache.tomcat.util.bcel.classfile.Constant.readConstant(Constant.java:97)
at org.apache.tomcat.util.bcel.classfile.ConstantPool.<init>(ConstantPool.java:54)
读取常量池时在Constant.readConstant方法中用switch-case对读到的首字节进行判断(这个字节标识了常量属于哪种类型),如果无法匹配则抛出异常,
(JAVA字节码 https://blog.csdn.net/umasanhao/article/details/47338623)
下面为反编译结果,异常抛出的值为19,在case的情况里没有定义所以抛出异常
static Constant readConstant(DataInputStream file) throws IOException, ClassFormatException {
byte b = file.readByte();
switch(b) {
case 1:
return ConstantUtf8.getInstance(file);
case 2:
case 13:
case 14:
case 17:
default:
throw new ClassFormatException("Invalid byte tag in constant pool: " + b);
case 3:
return new ConstantInteger(file);
case 4:
return new ConstantFloat(file);
case 5:
return new ConstantLong(file);
case 6:
return new ConstantDouble(file);
case 7:
return new ConstantClass(file);
case 8:
return new ConstantString(file);
case 9:
return new ConstantFieldref(file);
case 10:
return new ConstantMethodref(file);
case 11:
return new ConstantInterfaceMethodref(file);
case 12:
return new ConstantNameAndType(file);
case 15:
return new ConstantMethodHandle(file);
case 16:
return new ConstantMethodType(file);
case 18:
return new ConstantInvokeDynamic(file);
}
}
那么19是个啥情况呢?
JDK9新增了一个特别重要的特性——模块化,使用JDK9创建模块打包后有一个module-info.class的类
其中常量结构可能有19 - CONSTANT_Module and 20 - CONSTANT_Package两种,不在原有的11种取值之内
以下为相关的tomcat的bug报告,留言里讨论并说明了问题原因及修复方案
https://bz.apache.org/bugzilla/show_bug.cgi?id=60688
最后在如下版本做了修复
- trunk for 9.0.0.M18 onwards
- 8.5.x for 8.5.12 onwards
- 8.0.x for 8.0.42 onwards
- 7.0.x for 7.0.76 onwards
比较org.apache.tomcat.util.bcel.classfile.Constant.readConstant的代码确认,我正在使用的tomcat版本 7.0.61(未修复)和8.0.53(已修复)
修复部分的代码如下:
case 16:
case 19:
case 20:
skipSize = 2;
break;
case 9:
其他相关:
另外一种解决方案,通过修改tomcat加载的配置,不对相应的jar进行扫描
https://blog.csdn.net/JackRen_Developer/article/details/82288488
tomcat.util.scan.StandardJarScanFilter.jarsToSkip
jetty也有类似的问题,jetty新版本解决了加扫描module-info.class的问题,忽略了这个类
https://blog.csdn.net/baidu_34036884/article/details/80151963