JDK 的这3个bug,你发现了吗?

Wesley13
• 阅读 897

JDK 的这3个bug,你发现了吗?

作者:孤独的探索号

来源:my.oschina.net/tommylemon/blog/2967187

1.Annotation引用非空enum数组返回空数组

首次发现时的环境:JDK 1.8

首次发现所在项目:APIJSON

测试用例:

public enum RequestRole {  /**未登录,不明身份的用户   */  UNKNOWN,  /**已登录的用户   */  LOGIN,  /**联系人,必须已登录   */  CONTACT,  /**圈子成员(CONTACT + OWNER),必须已登录   */  CIRCLE,  /**拥有者,必须已登录   */  OWNER,  /**管理员,必须已登录   */  ADMIN;  //似乎不管怎么做,外部引用后都是空值。并且如果在注解内的位置不是最前的,还会导致被注解的类在其它类中import报错。  //虽然直接打印显示正常,但被@MethodAccess内RequestRole[] GET()等方法引用后获取的是空值  public static final RequestRole[] ALL = {RequestRole.UNKNOWN};//values();//所有  public static final RequestRole[] HIGHS;//高级  static {    HIGHS = new RequestRole[] {OWNER, ADMIN};  }  public static final String[] NAMES = {      UNKNOWN.name(), LOGIN.name(), CONTACT.name(), CIRCLE.name(), OWNER.name(), ADMIN.name()  };}@MethodAccess(    GETS = RequestRole.ALL,    HEADS = RequestRole.HIGHS    )public class Verify {}public class DemoVerifier {  // <TableName, <METHOD, allowRoles>>  // <User, <GET, [OWNER, ADMIN]>>    public static final Map<String, Map<RequestMethod, RequestRole[]>> ACCESS_MAP;  static { //注册权限        ACCESS_MAP = new HashMap<String, Map<RequestMethod, RequestRole[]>>();    ACCESS_MAP.put(Verify.class.getSimpleName(), getAccessMap(Verify.class.getAnnotation(MethodAccess.class)));  }  public static HashMap<RequestMethod, RequestRole[]> getAccessMap(MethodAccess access) {    if (access == null) {      return null;    }    HashMap<RequestMethod, RequestRole[]> map = new HashMap<>();    map.put(GET, access.GET());    map.put(HEAD, access.HEAD());    map.put(GETS, access.GETS());    map.put(HEADS, access.HEADS());    map.put(POST, access.POST());    map.put(PUT, access.PUT());    map.put(DELETE, access.DELETE());    return map;  }}

解决方案:

不抽象数组常量ALL,HIGHTS等,而是在每个用到的地方硬编码写死具体的值。

2.ArrayList可通过构造函数传入非指定泛型的List并在get时出错

首次发现时的环境:JDK 1.7

首次发现所在项目:APIJSON

测试用例:

JSONArray arr = new JSONArray(); //com.alibaba.fastjson.JSONArrayarr.add("s");List<Long> list = new ArrayList<>(arr); list.get(0); //Exception cannot cast String to Long

解决方案:

1.改用 Open JDK8

2.升级 JDK

注:后面多次测试,已无法复现。

3.基本类型在三元表达式内可赋值为null,编译通过但运行出错

首次发现时的环境: JDK 1.7

测试用例:

int i = true ? null : 0; //Exception in thread "main" java.lang.NullPointerException

首次发现所在项目:ZBLibrary

解决方案:

在给基础类型用3元表达式赋值时,null 先转为基础类型的默认值。

最后再提2个不是bug,但容易引发编程bug的问题:

1.局部变量和同名的全局变量能在一个方法内,编译通过,运行也正常。

public class Test {                int val;        @Override        public String toString() {            val = 1;            String val = "";            return super.toString();        }    }

如果两个变量中间隔了比较长的其它代码,很可能会导致开发人员将两者混淆,导致逻辑认知错误,从而写出或改出有问题的代码。

解决方案:

命名局部变量前先搜素,确保没有已声明的同名全局变量。

2. (非 JDK bug)Gson 通过 TypeToken 转换 List 能写入不属于 T 类型的数据,get 出来赋值给 T 类型的变量/常量报错。

String json = "[1, '2', 'a']";        Type type = new TypeToken<Integer>(){}.getType();        Gson gson = new Gson();        List<Integer> list = gson.fromJson(json, type);                Integer i = list == null || list.isEmpty() ? null : list.get(1); //Exception cannot cast String to Integer

解决方案:

1.手动检查列表内数据都符合泛型 T

2.改用 fastjson 等其它能静态检查类型的库。

欢迎在留言区说说你的看法,一起探讨交流。

结束

博主还有很优秀的技术交流群,很多技术大拿,CTO,活跃度百分八十以上。问题解答百分之90以上。加博主好友后回复【加群 】 ,然后回答技术问题,答对者才能进入,博其他广告主和商勿扰进群介绍,当然也会有一些学习资源,群里直接回复资源介绍

原创系列

数据结构+算法(第11篇)玩平衡二叉树就像跷跷板一样简单!

算法+数据结构(第10篇)叉堆“功夫熊猫”的速成之路


数据结构+算法(第09篇):菜鸟也能“种”好二叉树!

数据结构+算法(第08篇):史上最猛之递归屠龙奥义

数据结构+算法(第07篇),动态编程!黄袍加身!

=================================================================================================================================================================================================================================================================================================================================================================================================================================================================================================================================================================================================================================================

算法+数据结构(第06篇):再不会“降维打击”你就出了!

=====================================================================================================================================================================================================================================================================================================================================================================================================================================================================================================================================================================================================================================================

算法+数据结构(第5篇):小白也能玩转数组和链表啦!

算法+数据解构(第04篇),空间复杂度你真的懂了吗?

数据结构+算法(第03篇)KO!大O--时间复杂度

算法+数据结构(第02篇)玩扫雷就是优化算法

算法+数据结构(第01篇)走下神坛吧!算法


是不是很棒!分享一下吧,小伙伴们!

JDK 的这3个bug,你发现了吗?

本文分享自微信公众号 - Java研发军团(ityuancheng)。
如有侵权,请联系 support@oschina.cn 删除。
本文参与“OSC源创计划”,欢迎正在阅读的你也加入,一起分享。

点赞
收藏
评论区
推荐文章
blmius blmius
3年前
MySQL:[Err] 1292 - Incorrect datetime value: ‘0000-00-00 00:00:00‘ for column ‘CREATE_TIME‘ at row 1
文章目录问题用navicat导入数据时,报错:原因这是因为当前的MySQL不支持datetime为0的情况。解决修改sql\mode:sql\mode:SQLMode定义了MySQL应支持的SQL语法、数据校验等,这样可以更容易地在不同的环境中使用MySQL。全局s
皕杰报表之UUID
​在我们用皕杰报表工具设计填报报表时,如何在新增行里自动增加id呢?能新增整数排序id吗?目前可以在新增行里自动增加id,但只能用uuid函数增加UUID编码,不能新增整数排序id。uuid函数说明:获取一个UUID,可以在填报表中用来创建数据ID语法:uuid()或uuid(sep)参数说明:sep布尔值,生成的uuid中是否包含分隔符'',缺省为
待兔 待兔
6个月前
手写Java HashMap源码
HashMap的使用教程HashMap的使用教程HashMap的使用教程HashMap的使用教程HashMap的使用教程22
Jacquelyn38 Jacquelyn38
3年前
2020年前端实用代码段,为你的工作保驾护航
有空的时候,自己总结了几个代码段,在开发中也经常使用,谢谢。1、使用解构获取json数据let jsonData  id: 1,status: "OK",data: 'a', 'b';let  id, status, data: number   jsonData;console.log(id, status, number )
Stella981 Stella981
3年前
KVM调整cpu和内存
一.修改kvm虚拟机的配置1、virsheditcentos7找到“memory”和“vcpu”标签,将<namecentos7</name<uuid2220a6d1a36a4fbb8523e078b3dfe795</uuid
Stella981 Stella981
3年前
Android So动态加载 优雅实现与原理分析
背景:漫品Android客户端集成适配转换功能(基于目标识别(So库35M)和人脸识别库(5M)),导致apk体积50M左右,为优化客户端体验,决定实现So文件动态加载.!(https://oscimg.oschina.net/oscnet/00d1ff90e4b34869664fef59e3ec3fdd20b.png)点击上方“蓝字”关注我
Wesley13 Wesley13
3年前
00:Java简单了解
浅谈Java之概述Java是SUN(StanfordUniversityNetwork),斯坦福大学网络公司)1995年推出的一门高级编程语言。Java是一种面向Internet的编程语言。随着Java技术在web方面的不断成熟,已经成为Web应用程序的首选开发语言。Java是简单易学,完全面向对象,安全可靠,与平台无关的编程语言。
Stella981 Stella981
3年前
Django中Admin中的一些参数配置
设置在列表中显示的字段,id为django模型默认的主键list_display('id','name','sex','profession','email','qq','phone','status','create_time')设置在列表可编辑字段list_editable
Wesley13 Wesley13
3年前
MySQL部分从库上面因为大量的临时表tmp_table造成慢查询
背景描述Time:20190124T00:08:14.70572408:00User@Host:@Id:Schema:sentrymetaLast_errno:0Killed:0Query_time:0.315758Lock_
Python进阶者 Python进阶者
1年前
Excel中这日期老是出来00:00:00,怎么用Pandas把这个去除
大家好,我是皮皮。一、前言前几天在Python白银交流群【上海新年人】问了一个Pandas数据筛选的问题。问题如下:这日期老是出来00:00:00,怎么把这个去除。二、实现过程后来【论草莓如何成为冻干莓】给了一个思路和代码如下:pd.toexcel之前把这