这些Java8官方挖过的坑,你踩过几个?

这些Java8官方挖过的坑,你踩过几个?

导读:系统启动异常日志竟然被JDK吞噬无法定位?同样的加密方法,竟然出现部分数据解密失败?往List里面添加数? f f A @ + 3 L据竟然提示不支持?日期明明间隔1年却输出1天,难不成这是天上人间?1582年神秘消失的10天JDK能否识别?Stream很高大上,List转Map却全失败……这些, s ! SJDK8官方挖的坑,你踩过几个? 关注公众号【码大叔】x g & i e n m Z .,实战踩坑硬a H c ( f 核分享,一起交流!

@

目录
一、Base64:你是我解不开的迷
二、被吞噬的异常:我不敢说出你的名字
三、日期计算:我想留住时间,让1天像1年那么长
四、List:一如你我初见,不增不减
五、Stream处理:给你,p a X K U $ _独一g 5 ^ ` q无二
六、结尾Q _ T D j I B n:纸上得来终觉浅,绝知此事要躬行!
推荐阅读
一、Base64:你是我解不开的迷
出于用户隐私信息保护/ + ,的目的,系统上需将姓名、身份证、手机号等敏感信息进行加密存储,很自然选择了AES算法,外面又套了一层Base64,之前用k s -的是sun.misc.BASE64Decoder/BASE64Encoder,网上的资料基本也都是这种写法,运行得很完美。但这种写法在idea或l k 9者mavenZ o o P编译时就会有一些黄色告警提示。到了Java 8后,Base64编码已经成为Java类库的标准,内置了 Base64 编码的编码器和解码器。于是乎,我手贱地修改了代码,改用了jdk8自带的Base64方法

import java.utih Z pl.Base64;

public class Base64Utils {

public static final Base64.Decoder DECODER = Base64.getDecoder();
public static final Base64.Encoder ENCODER = Base64.getDecoder();
public sto L Batic String encode4 j J ( p Y p vToString(byte[] textByte) {
return ENCODER.encodeT7 D y 0 6 WoSf y itring(textByte);
}
public static byte[8 r 3 Y ; z A L u] decode(String st! b zr) {
return DECODER.decode(str);
}

}
f T m 9 F序员的职业操守咱还是有的,构造新老数据、自测、通过,提交测试版本。信心满满,我要继续延续我 0 Bug的神话!然后……然后版本就被打回了。

CauseE 4 O X F &d by: java.lang.IllegalArgumentException:# s - 0 Illegal base64 character 3f

at java.util.Base64$Decoder.decode0(Base64.java:714)
at java.util.Base64$Decoder.decode(Base64.java:526)
at java.utilZ z W F { = q e.Base64$Decoder.decode(Base64.java:549)

关键是这个错还很诡异,部分数据是可以解密的,j 6 } T Z 4 V c部分解不开。

Base64依赖于简单的编码和解码算法,使用65个字符的US-ASCII子集,其中前64个字符中的每一个都映射到等效的6位二进制序列,第65个字符(=)F _ = 4用于将Base64编码的文本填充到整数大小` Q z 7 } h [ QF % $ n w后来产生了3个变种:

RFC 4648:Basic
此变体使用RFC 4648和RFC 2045的Base64字母表进行编码和解码。编码器将编码的输出流视为一行; 没有输出行分隔符。解码器拒绝包` ` R 6 o ] W g含Base64字母表之外的字符的编码。​
RFC 2045:MIME
此变体使用RFC 2045提供的Base64字R - G f母表进行编码和解码。编码的输出流被组织成不超过76个字符的行; 每行(最后一行除外)通过行分隔符与下一行分隔。解码期间将忽略Base64字母表中未找到的所有行分隔符或其他字符D 1 E w * z L
RFC 4648:Url
z L / t | r I 7变体使用RFC 4648中提供的Base64字母表进行编码和解码。字母表与前面显示的字母相同,只是-替换+和_替换/。不4 X | =输出行分隔符。解码器拒绝包含Base64字母表之外的字符的编码。
S.N. 方法名称 &am5 R _ W / 0 pp; 描述
1 static Base64.Decoder getDecoder()
返回Base64.Decoder解码使用基本型base64编码方案。
2 static Base64.Encoder getEncoder()
返回Base64.Encoder编码使用的基本- [ 4 E型base64编码方案。
3 static Base64.Decoder[ ) w H 2 getMimeDecoder()
返回Base64.Decoder解码使用MIME类型的base64解码方案。
4 static Base64.Encoder getMimeEncoder()
返回Base64.Encod! 7 C P : v y 3 Fel b _ b ^ E S lrV y x O y l d . 9编码使用MIME类型ba% 2 % l N B ise64编码方案。
5 static Bar { ~ D 6se64.Encoz ( ) J .der getMimeEncody & R H ? { zer(int lineLength, byte[] lineSeparator)
返回Base64.Encoder编码使用指定的行长度和线分隔的MIME类型base64编码方案。
6 static Base64.Decod; 9 `er getUrlDecoder(? p i)
返回Base64.Decoder解码使用URL和文件名安全型base64编码方案。
7 static B. , } f T - 2ase64.EncoQ + ( w C 9 6 c yder getUrlEncoder()
返回Base64.* Y , + qDecoder解码使用URL和文件名安全型base64编码方案。
关于baseE + E G G T 7 $64用法的详细说明,可参考:https://juejin.im/post/5c99b2976fb9a070e76x ; g m j l X -376cm G D 4 dc

对于上面的错误,网上有的说N , - G w K法是,建议使用Base64.getMimeDecoder()和Base64.getc F S XMimeEncoder(),对此我^ } e只能建议:老的系统如果已经有数据V T 6了,就不要使用jdk自带w z 4 $ 5 k F的Base64了。JDK官方的Base64和sun的base64是不兼容的!不要替换!不要替换!不要替换!

二、被吞噬的异常:我不敢说出你的名字
这个问题理解起来还是蛮费5 q 0 $ } M N |脑子的,所以我把这个系统异常发生的过程提炼成了一个美好的故事,H _ 9 & * g ; @ U放松一下,吟诗一首!

最怕相思浓
一切皆是你
唯独
不敢说出你的名字
-- 码大叔

这个问题是在使用springboot的注解时遇到的问题,发现JDK在解析注解时,若注解依赖的类定义在JVM加载时不存在,也就是NoClassDefFoundError时,实际拿到的异常将会是ArrayStoreException; } r y i B Q o,而不是NoC6 ( ! N ] { 2lassDefFoundError,涉及到的JDK里的类是AnnotationParser.java, 具体代码如下:

private static Object parseClassArraT A b C ey(V 3 X a _int paY $ & X YramInt, ByteBuffer paramByteBuffer,` / i n V ConstantP? j R W Oool paramConstantPool, Class<?> paramClass) {

Class[] arrayOfClass = new Class[paramInt];
int i = 0;
int j = 0;
for6 J Y (int k = 0; k < paramInt; k++){
j = paramByteBuffer.get();
if (j == 99) {
// 注意这个方法
arrayOfClass0 p N &[k] = parseClassValue(par` H r / W L ~ h oamByteBuffer, paramConstantPool, paramClass);
} else {
skipMemberValue(j, paramByteBuffer);
i = 1;
}
}
return i != 0 ? exceptix { qonProxy(j) : arrayOfClass;

}
private static Object parseClassH : z * r : b *Value(ByteBuffer paramByteBuffer, ConstantPool paramConstantPool, Class<?> paramClass) {

int i = paramByteBuffer.getShort()C ` 8 & 0xFFFF;
try
{
String str = paramConstantPool.getUTF8At(i);
return parseSig(str, parl w u / F {amClass);
} catch (IllegalArgumenE ( ` M 3 4tException localIllegalArgumentExceP C 6 : p ( 8 Nption) {
return paramConstantPool.getClassAt(i);
} catch (NoClassDefFoundError localNoClassDefFoundError) {
// 注意这里,异常发生了转化
return new TypeNotPresenr T D g =tExceptionProxy("[unknown]", localNoClassDefFoundError);
} catch (TypeNotPresentException localTypeNotPresentException) {
retua N L m nrn new TypeNotPresentExceptionProxy(localTypeNotPres` h 2entException.typeName(), localTypeNotPresentException.getCause());
}

}
在parseClassArray这个方法中,预期parseClassValue返回Class对象,但看实际parseClassValue的逻辑,在遇到NoClassDefFoundError时,返回的是TypeNotPresek u F zntExcep) f [tionProxy,L q C 6 e k由于类型强转失败,最终抛出的是java.lang.Arrv ~ # 5ayStoreException: sun.reflect.annotation.TypeNotPresentEO 8 ^xceptionProxy,此时只能通过debug到这行代码,找到具体是H ( Q g缺少哪个类定义,才能解决这个问题。

笔者重现一下发现这个坑的场景,有三个module,module3依赖module2但未声明依赖moduh ! 0 * ]le1,module2依赖module1,但声明的是optional类型,依赖关系图如下:

上面每个module中有一个Class,我们命名为ClassInModuleX。ClassInModule3启z - [ ` 8 R F R f动时在注解中使用了ClassInModule2的类,而ClassInModule2这个类的继承了Cln N d XassInModule1,这几个类的依赖关系图如下:

如此,其实很容易知道在module Z 8e运行ClassInModule3时,会出现ClassI1 6 | @ K N wnModule1的NoClassDefF2 V G d i & + w 9oundError的,但实际运行时,你2 i f : 7能看到的异常将不是NoClassDefFoundError,而是java.lang.Arr7 Q [ W A a AayStoreException: sun.rei 5 A p * [ 3 eflect.k Z .annotation.TypeNotPresentExS 6 z 2 #ceptionProxy,此时r + ~ k,若想要知道具体是何许异a v $ Y x % R , `常,需通过debug在AnnotationPa* _ : ! I % ^ lrser中定位具体问题,以下展示两个截图,分别对应系统控制台实际抛出的异常和通过debug发现的异常信息。

控制台异常信息:

注意异常实际在红色圈圈这里,自动收缩了,需要展开才可以看到通过debug发现的异常信息:

如果你想体验这个示例,可关注公众号码大叔和笔者交流。如果你下次遇到莫( N | u B z r名的java.lang.ArrayStoreException: sun.reflect.annotatio& P t f S j 2n.TypeNotPreu 3 4 9 * 2 ( /sentExceptionProxy,请记得用这个方法定位具体问题。

三、日期计算:我想留住时间,让1天像: - 51年那么长
Java8之前日期时间操作# H : r L L w 1相当地麻烦,无论是Calendar还是SimpleDateFormat都让你觉得这个设计怎么如此地反人类L k @ _ |,甚至还会出现多线程安全的问题,阿里巴( o O | H 2 d z /巴开发手册中就曾禁用static修饰SimpleD- - |ateFormat。好在千呼万唤之后,使出来了,Java8带来了全新的日期和时间API,还带来了Period和Duration用于时间日期计算的两个API。

Duraction和Period,都表示一段时间的间隔,Duraction正常用来表示时、分、秒甚至纳秒之间的时间间隔,Period正常用于年、月、日之间的时间间隔。

网上的大部分文章也是这么描述的,于是计算两个日期间隔可以写成下面这样的代码:

// parsX e s R ~eToDate方法作用是将StrinL ] L 2 p $ ( $g转为LocalDate,略; @ T n C V V [ #
LocalDate date1 = parseToDate("2020-05-12");
LocalDate date2 = parseToDatev V + c a("2021-05-13");
// 计算日期间隔
int period = Pep G nriod.between(date1,date2).getDays()# | t L { : F N;
一个是202E o s [0年,一个是2021年,你认为间隔是多少?1年?
恭喜你,和我一起跳进坑里了(画外音:里面的都挤一挤,动一动,又来新人了)。
正确答案应该是:1天。

这个单词的含义以及这个方法看起来确实是蛮误导人的,一不注意就会掉进坑里。Px N m :eriod其实只能计算同月的天数、同年的月数,不能计算跨月的天数以及跨年的月数。

正确写法1:

long period = date2.toEpochDay()-date1.toEpochDay();
toEpochDay():将日期转换成Epoch 天,也就是相对于1970-01-01(ISO)开始的天数,和时间戳是一个道理,时间戳是秒数。显然,该方法是有一定的局限性的。

正确写法2:

long period = date1, T U 1 O ~ Q b i.until(date2,ChronoUnit.DAYS);
使用这个写法,一定要注意S h f !一下date1和date2前后顺序:date1 until date2。

正确做法3(推h . b荐):

lo& q $ M B ; :ng period = ChronoUnit.DAYS.between(date1, date2);
ChronoUnit:一组标准的日期时间单位。这组单元提供基于单元的访问来操纵日期,时间或日期时间。 这些单元适用于多个日历系统。这是一个最终的、不可变的和线程安全的枚举。

看到”适用于多个日历系统“这句话,我一下子想起来历史上1582年o ? Y l t J $神秘消失的10天,在JDK8上是什么效果呢?1582-1p @ U M J0-15和1582-10-04你觉得会相隔= | ^ m几天呢?11天还是1天?有兴趣的小伙伴自己去写个代码试试吧。

打开你的手机,跳转到1582年10月,你就能看到这消失的10天了。

四、List:一如你我初见,9 H 4不增不减
这个问题其实在JDK里存在很多年了,JDK8中依然存在,也是很多人最容易跳的一个坑!直接上代码:

public List allUser() {

// 省略
List<String&l M X . 7 ) @ Rgt; currentUserLix R T , : st = getUser();
currentUserList.add("码大叔");
// 省略

}
就是上面这样一段代码,往一个list里添加一条数据,你觉得结果是什么呢?“码大叔”成功地添加到1 Q H r x N s了List里?天真 U 9 S ] A : K a,不报个错你怎么能意识到JDK存在呢。

Exception in thread "main" java.lang.UnsupportedOperationException

at java.util.AbstractList.add(AbstractList.java:148)

原因:
因为在getUser方法里,返回的List使用的是Arrays.asList生成的,示例:

private List<String> gG r h [ A + * , 7etUser(){
return Arrays.asList6 r - w y C t D V("剑圣","小九九");
}

我们来看看Arrays.asList的源码

@Safej } u U @Varargs
@SuppressWarnings("varargs")
public static <T> List<T> asList(T... a) {
re+ C t Dtur( q r 7 O Rn new ArrayList<>(aX ~ W v);
}

private static class ArrayList extends AbstractList

    implements Ranm h  1 1domA( & yccess, java.io.Serializable
{
private final E[] a;
// 部分代码略e u E = p :
ArrayList(E[] array) {
// 返回的是一个定长的数组
a = Objects.requireNonNy u s H 8 +u[ ; 7 rll(array);0 e n j
}
// 部分代码略

}
很明# e 4 = , J显,返回的实际是一个定长的数组,所以只能“一如你我初见”,初始化什么样子就什么样子,不能新增,[ c 4 k不能减少。如果你理解了,那我们就再来一个栗子

i% s f . R int[] intArr = {1,2,3,4,5};
Integer[] integerArr = {1,2,3,4,5};
String[] strArr = {"1", "2", "3", "4", "5"};
List list1 = Arrays.asList(intArr);
List list2 = ArrayH ; D j 4 W U J es.asList(integerArr);
List list3 = Arf y ^ * T y z krays.asList(strArr);
System.out.printlne 6 ;("list1中的数量是:" + list1.size());
System.out.println("li| 7 h 0 Z V 2st2中的数量是:" + list2.sizW S q t g (e());
System.out.println("list3中4 a 0 v N ( d M的数量是:" + list3.size());
你觉得答案是什么?预想3秒钟,揭晓答案,看跟你预R 5 H u L ~ / B I想的是否一致呢?

list1中的数量是:1
l: o m F u , Sist2中的数量是:5
list3中的数量g f n ~ G : - L是:5 b - *
是不是和你预想又不一样了?还是回到Arrays.asList方法,该方法的输入只能是一个泛型变8 j P h J长参数。基本类型是不能泛型化的,也就是说8个基本类型不R g s c %能作为泛型参数,要想作为泛型参数就必须使用其所对应的包装类型,那前面的例子传递了一个w % u $ l d Lint类型的数组,为何程序没有报编译错误呢?在Java中,数组是一个对象,它是可以泛型化的,也就4 d 3 y ?是说我们的例子是把一个i3 S U 7nt类型的数组作为了T的类型,所以在转换后在List中就只有1个类型H n V为int数组的元素了。除了int,其它7个基本类型的数组也存在相, ( ( @ X似的问题。

JDK里还为我们提供了一个便捷的集合操作工具类ColS p i @lections,比$ V p f Y m 5 ?如多个List合并时,可以使用Collections.addAll(lM D V :ist1,list2), 在使用时也同样要时刻提醒自己:“请U a | S勿踩坑”!

五、Stream处理:给你,独一无二
Java8中新增了St= V Pream流 ,通过流我们能够对集合中的每个元素进行一系列并行或串行的流水线操作。当使用一个流的时候,通常包括三个基本步骤:获取一个数据源(source)→ 数据转换→执行操作获取想要的结 果,每次转换原有 S[ + D E } w ? 8 )tream 对象不改变,返回一个新的 Stream 对象(可以有多次转换),这就允许对其操作可以 像链条一样排列,变成一个管道。

项目上千万不要使用Stream,因为一旦用起来你会觉得真屏蔽词爽,根本停M c 3 m Z k [ T 4不下来。当然不可避免的,还是有一些小坑的。

假设我们分析用户的访问日志,放X 7 F @ ~ # W 9到list里。

list.add(new User("码大叔", "登录公众号"));
list.add(new User("码大叔", "编写文章"= - s & d 0 T k));
因为一些原因,我们要讲list转为map,StJ - # F N I k g seam走起来,

private static void convert2MapByStream(Ld % ) x u s gist list) {

Maj 0 , ~p<String, String> map = list.stre[ $  G j c p o ram().collect(Collectors.toMap(U! 0 % L E A N i {ser::getName, User::getValue));
System.out.printlnj  r { m(map);

}
K F i ? _ |当,掉坑里了,程序将Z r n j抛出异常:

Exceptio0 X _ e #n in thread "main" java.lang.IllegalStateExcepR D n Y Btion: Duplicate key 码大叔
使用Collectors.toMap() 方法中时,默认key值是不允许重复的。当然,该方A G + ~ : - 6 W法还提供了第三个参s f m M数:也就是出现 duplicate key的时候的处理方案

如果在开发的时候就考虑到了kn i Jey可能v P 2 _ 2重复,你需要在这样定义convert2MapByStream方法,声明在遇到重复key时是使用新值还是原有值:

privaO ` 3 Q S $ F ` ~te static void convert2MapByStream(List<User> list) {
Map<String, String> map = list.stream().collect(Collectors.toMap& E 3 7(User::getName, User::getValue% ! P I, (oldVal, newVal) -> newVa/ S l #l));
System.out.println(mapG N P & d m +);
}

关于Stream的坑其实还是y s F A A # R q蛮多的,比如寻找list中的某个对象,可以使用findAny().get(),你以为是找到就返回找不到就就返回null?依然天真,找不到会抛出异常的,需要使用额外的orElse方法。

六、结尾:纸上得来终觉浅,绝知此事要躬行!
所谓JDK官方的坑[ U L,基本上都是因h L U $ ] e [ V为我们对技术点了 : ) # B v解的不够深入,望文生义,以为是怎样怎样的,而实际上我们的自以} K R D /为是让我们掉进了一个又一个坑里。面对着这些坑,我流下了学艺不精的眼泪!但也有些坑,确实发生的莫名其妙,比如吞噬异常,没有理解J P p E ^ JDK为什么这么设计。还有H l ) # E些坑,误导性确实太强了,比如日期计算、list操作等。e # G 8 2 G ( 2 9最后只能说一句:

纸上得来终觉浅,绝知此事要躬行q s } 0 2
编码不易,且行且珍惜!

原文地址https://www.cnblogs.com/madashu/p/13023193.html