包装类这颗语法糖,其实并不甜

2023-11-12

历史文章推荐:

  1. 你真的了解时间吗
  2. 细数ThreadLocal三大坑,内存泄露仅是小儿科
  3. Java 8 ConcurrentHashMap源码中竟然隐藏着两个BUG
  4. ConcurrentHashMap中有十个提升性能的细节,你都知道吗?
  5. HashMap面试,看这一篇就够了
  6. 七种方式教你在SpringBoot初始化时搞点事情

包装类在Java 5中和泛型一起引入,引入包装类的原因有两点:

  1. 解决无法创建基本类型泛型集合的问题
  2. 加入对基本类型为null这个语义的支持

并提供boxingunboxing的语法糖,让编译器支持基本类型和包装类的自动转化,减少开发者的工作量。但是经常有同学因为误用包装类导致惨烈的线上问题,在使用包装类的时候务必需要注意一下四点:

  1. 与基础类截然不同的==equals语义
  2. 糟糕的性能
  3. 不易察觉的NPE问题
  4. 令人疑惑的API设计

1. 相等还是不相等?这是个问题

比如以下代码片段

class Biziclop {
    public static void main(String[] args) {
        System.out.println(new Integer(5) == new Integer(5)); // false
        System.out.println(new Integer(500) == new Integer(500)); // false

        System.out.println(Integer.valueOf(5) == Integer.valueOf(5)); // true
        System.out.println(Integer.valueOf(500) == Integer.valueOf(500)); // false
    }
}

第一个和第二个语句返回false是比较容易理解的,因为对于Java中的对象调用=其实是在比较对象在堆上的地址,由于两个对象都是新建的,所以地址肯定不等,返回false。比较令人疑惑的是第三个语句,按照我们前面的分析,应该也返回false才对,但其实Integer.valueOf(5) == Integer.valueOf(5)比较的结果是true,这是因为JVM缓存了-128-127的整数,所以当数值在这个区间的时候,返回的对象都是同一个的。第四个语句因为数值已经不再-128-127的区间范围,所以返回了false

上面的这几个例子都是比较经典的例子,大家比较熟悉,一般也比较难掉坑里,但是下面的几个例子就比较有迷惑性了

class Biziclop {
    	public static void main(String[] args) {
        List<Long> list = new ArrayList<>();
        list.add(Long.valueOf(200));
        System.out.println(list.contains(200)); // false
        
        Long temp = 0L;
				System.out.println(temp.equals(0)); // false
       System.out.orintln(0==0L); // true
	}
}

原因在于

public boolean equals(Object obj) {
    if (obj instanceof Long) {
      return value == ((Long)obj).longValue();
    }
    return false;
}

包装类重写了equals方法,导致包装类即便是调用equals方法比较大小,也会和基本类型出现不一致的结果。与基础类截然不同的==equals语义经常会导致代码走到非期望的分支,再配上JVM对数字独特的缓存策略,极容易出现测试环境和正式环境不一样的运行结果。

2. 糟糕的性能

Effective Java》中有如下的例子:

public static void main(String[] args) {
    Long sum = 0L; // uses Long, not long
    for (long i = 0; i <= Integer.MAX_VALUE; i++) {
        sum += i;
    }
    System.out.println(sum);
}

这段代码的耗时比使用基本类型long的版本慢6倍(声明变量sum 类型为 Long 的耗时是43秒, 如果声明变量sum为基本类型long,则耗时6.8秒)。导致这样的原因是包装类要经过在堆上开辟内存空间,初始化,内存寻址以及数据载入寄存器的过程,性能差也就不足为奇了。因此Joshua Bloch对开发者的建议是:Avoid creating unnecessary objects.

在经典的JMH workbench上跑的包装类和基础类性能对比如下图所示:
在这里插入图片描述

可以看到与基础类相比,包装类普遍要慢不少。

图片来源地址:https://www.baeldung.com/java-primitives-vs-objects

3. 不易察觉的NPE

不同于基本类型,作为对象的包装类是可能为null的,这就意味着一个指向null的包装类unboxing的时候会抛出NPE异常,比如以下代码:

Integer in = null;
...
...
int i = in; // NPE at runtime

这段代码也是比较明显的,但是如果包装类遇到三元运算符,则会出现更复杂的NPE

class Biziclop {
    public static void main(String[] args) {
      Boolean b = true ? returnsNull() : false; // NPE on this line.
      System.out.println(b);
    }
  	public static Boolean returnsNull() {
    	return null;
		}
}

这跟Java中三元运算符类型的判定有关系,有一条判定规则是,

如果三元运算符的第二个或者第三个参数是基本类型T,并且另一个是相应的包装类型的话,那么三元运算符的返回类型就是这个基本类型T

所以在上面的代码中,returnsNull的返回值还要进行一次unboxing,因此抛出了NPE.

4. 令人疑惑的API

Long这个类中,有一个apigetLong,其声明如下:


    /**
     * Determines the {@code long} value of the system property
     * with the specified name.
     */
    public static Long getLong(String nm) {
        return getLong(nm, null);
    }

这个api的作用是获取JVM中的属性值的,并且转换为Long 类型,比如:

class Biziclop {
    	public static void main(String[] args) {
        System.setProperty("22", "22");
        System.setProperty("23", "hello world!");
        System.out.println(Long.getLong("22")); // 22
        System.out.println(Long.getLong("23")); // null
        System.out.println(Long.getLong("24")); // null
		}
}

这个api的设计妥妥是一个反例,经常有同学误用,把它当成Long.valueOf或者是Long.parseLong,结果返回不符合期望的值。

5. 最佳实践

《阿里巴巴Java编程手册》对包装类的使用有以下三条建议:

  1. 所有POJO类属性使用包装类
  2. RPC方法的返回值和参数使用包装类
  3. 所有的局部变量使用基本数据类型

说明:POJO类属性没有初值是提醒使用在在需要使用时,必须自己显式的进行赋值,任何NPE问题,或者入库检查,都有使用者来保证。

正例:数据库的查询结果可能是null,因为自动拆箱,用基本数据类型接受有NPE的风险

反例:某业务的交易报表上显式成交额涨跌情况,即x%,x为基本数据类型,调用的HSF服务,调用不成功时,返回的是默认值,页面展示0%,这是不合理的,应该展示成中划线-,所以包装类的null值,能够表示额外的信息,如:远程调用失败,异常退出。
在这里插入图片描述

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

包装类这颗语法糖,其实并不甜 的相关文章

  • Maven 2:如何将当前项目版本打包在WAR文件中?

    我正在使用 Maven 2 构建我的 Java 项目 并且正在寻找一种向用户呈现 pom xml 当前版本号的方法 例如使用 Servlet 或 JSP 据我所知 最好的方法是 Maven 将版本号作为文本文件打包到 WAR 中 这使我能够
  • 具有默认值的 Java JAX-RS 自定义参数

    假设我有这个 这只是一个示例 GET Path value address Produces application json public Response getAddress QueryParam user User user 用户是
  • JavaFX 图像未在舞台中显示

    我尝试了很多次 尝试了很多方法 但都无法让自己的形象在舞台上如我所愿 我认为这可能与java寻找资源的路径有关 但我不确定 因为我刚刚开始使用视觉库 在本例中为JavaFX 这是我的目录结构 MyProject assets img myI
  • 在Java中使用命令行编译多个包

    您好 我一直在使用 IDE 但现在我需要从命令行运行和编译 问题是我有多个软件包 我试图找到答案 但没有任何效果 所以我有 src Support java files Me java files Wrapers java files 你知
  • 在哪里可以获得有关 Java FitNesse 和 Slim 的一些教程? [关闭]

    就目前情况而言 这个问题不太适合我们的问答形式 我们希望答案得到事实 参考资料或专业知识的支持 但这个问题可能会引发辩论 争论 民意调查或扩展讨论 如果您觉得这个问题可以改进并可能重新开放 访问帮助中心 help reopen questi
  • Java 正则表达式 - 字母数字,最多一个连字符,句点或下划线,七个字符长

    我是 Java 正则表达式工具的新手 尽管它们潜力巨大 但我很难完成这项任务 我想编写一个正则表达式来验证遵循以下语法的输入字符串 小写字母和数字的任意组合 仅一个下划线 一个破折号或一个句号 无其他特殊字符 最小长度为 5 我想出了以下解
  • RMI 中的引用传递问题? [复制]

    这个问题在这里已经有答案了 有人可以告诉我我错在哪里 为什么这个 RMI 聊天应用程序不起作用 目标是通过远程对象或序列化对象实现客户端 服务器和逻辑之间的解耦 import javax swing import java awt even
  • java.lang.LinkageError:尝试重复的类定义

    为什么会发生错误以及如何修复它 02 13 02 pool 4 thread 2 WARN Exception in thread pool 4 thread 2 02 13 02 pool 4 thread 2 WARN java lan
  • 如何拦截 REST 端点以接收所有标头?

    我当前的代码是 Path login RequestScoped public class LoginResource GET SecurityChecked public Response getUser HeaderParam AUTH
  • 字符串池可以包含两个具有相同值的字符串吗? [复制]

    这个问题在这里已经有答案了 字符串池可以包含两个具有相同值的字符串吗 String str abc String str1 new String abc Will the second statement with new operator
  • JTable 和 JScrollpane 大小的问题

    我有一个JScrollPane with a JTable在里面 在里面JTable我最初有 3 行 稍后添加行 默认JTable我的 3 行很难看 因为JScrollPane calls getPreferredScrollableVie
  • Java-如何将黑白图像加载到二进制中?

    我在 FSE 模式下使用 Java 和 swing 我想将完全黑白图像加载为二进制格式 最好是二维数组 并将其用于基于掩码的每像素碰撞检测 我什至不知道从哪里开始 过去一个小时我一直在研究 但没有找到任何相关的东西 只需将其读入Buffer
  • Intellij 中的 Google OR-Tools:UnsatisfiedLinkError

    我正在建立一个应该使用 Google OR Tools 的 java 框架 下面的代码编译成功 但在运行时抛出异常 Exception in thread main java lang UnsatisfiedLinkError com go
  • 异步迭代器

    我有以下代码 while slowIterator hasNext performLengthTask slowIterator next 由于迭代器和任务都很慢 因此将它们放入单独的线程中是有意义的 这是对迭代器包装器的快速而肮脏的尝试
  • 如何将 arraylist 从 servlet 传递到 javascript?

    我通过在属性中设置数组列表并将其转发到 jsp 来从 servlet 传递数组列表 Servlet ArrayList
  • 改变for循环的顺序?

    我遇到一种情况 我需要根据用户输入以不同的顺序循环遍历 xyz 坐标 所以我是 3D 空间中的一个区域 然后是一组像这样的 for 循环 for int x 0 x lt build getWidth x for int y 0 y lt
  • 警告:无法更改每个人的权限:

    当运行 Java 快速入门示例时https developers google com drive web quickstart java hl hu https developers google com drive web quicks
  • 无法使用 wget 在 CentOS 机器上安装 oracle jdk

    我想在CentOS上安装oracle java jdk 8 我无法安装 java jdk 因为当我尝试使用命令安装 java jdk 时 root ADARSH PROD1 wget no cookies no check certific
  • 防止Java实例化的正确方法[关闭]

    就目前情况而言 这个问题不太适合我们的问答形式 我们希望答案得到事实 参考资料或专业知识的支持 但这个问题可能会引发辩论 争论 民意调查或扩展讨论 如果您觉得这个问题可以改进并可能重新开放 访问帮助中心 help reopen questi
  • 为什么范围为“provided”的依赖项会隐藏 Maven 中的传递依赖项?

    我的 Maven 项目中有三个模块 这稍微简化了 model包含JPA注释的实体类 坚持实例化一个实体管理器并调用它的方法 应用创建类的实例model 设置一些值并将它们传递给坚持 model and 坚持显然取决于javax persis

随机推荐

  • 权限认证。。

    链接 手摸手 带你用vue撸后台 系列二 登录权限篇 掘金 juejin cn 前端权限控制 一 前端权限管理及动态路由配置方案 ONEO阿喔哟的博客 CSDN博客 检查员工是否具有特权 param requestTokenBO 请求令牌B
  • 如何快速算出一个数有多少个因子(c++)

    如何快速算出一个数有多少个 多少种 因子 c int count int n int sum 1 for int i 2 i i lt n i if n i 0 int tmp 0 while n i 0 n i tmp sum sum t
  • Piecewise混沌映射/PWLCM混沌映射(含MATLAB代码)

    一 Piecewise混沌映射 PWLCM混沌映射 混沌映射是生成混沌序列的一种方法 常见的混沌映射方式有 Logistic映射 Tent映射 Circle映射 而 Piecewise映射作为混沌映射的典型代表 数学形式简单 具有遍历性和随
  • python文件操作(with open)——读取行、写操作

    一 基础语法 1 打开文件 这里只介绍一种常用方式 但是打开文件方式有很多种 掌握一种最适合自己的即可 推荐使用这种方式 因为不需要close 具体原因往下看 看到示例就懂了 打开文件的模式有很多种 r 读 w 写等 此处不做详细介绍 采用
  • sublime text3取消自动换行!

    菜单栏中取消view gt word wrap的勾选也可以取消其代码的自动换行 菜单栏选择preferences gt Setting User中添加 word wrap false 即可
  • 膜拜(离散化差分模板题)

    题目描述 小鱼有 n 名优秀的粉丝 粉丝们得知小鱼将会在一条直线上出现 打算去膜他 为了方便 粉丝们在这条直线上建立数轴 第 i 名粉丝有一个侦查区间 li ri 如果小鱼在 j li j ri 处出现 这名粉丝将立刻发现并膜他 小鱼希望膜
  • python 3 中文URL编码转换问题

    链接里面含中文 转成URL编码 先引入模块 from urllib request import quote gt gt gt ff 摄像头 gt gt gt ff quote ff gt gt gt ff E6 91 84 E5 83 8
  • sql 还原数据库 错误3154

    在SQL Server2005及以下版本做数据库备份还原时 需要首先建立数据库 然后才能进行数据库还原操作 而在SQL Server2005以上版本做数据库还原时 不需要建立数据库 可以直接进行数据库还原操作 否则执行数据库还原操作时会报3
  • 求阶乘之和(循环版)(利用阶乘函数)

    请编写函数 用循环方法求阶乘之和 SumFac n 0 1 2 3 n include
  • uniapp uview 登录页

  • DETRs Beat YOLOs on Real-time Object Detection论文详解

    论文题目 DETRs Beat YOLOs on Real time Object Detection 论文地址 https arxiv org abs 2304 08069 论文代码 mirrors facebookresearch Co
  • jmeter:linux环境运行jmeter并生成报告

    是一个java开发的利用多线程原理来模拟并发进行性能测试的工具 一般来说 GUI模式只用于创建脚本以及用来debug 执行测试时建议使用非GUI模式运行 这篇博客 介绍下在linux环境利用jmeter进行性能测试的方法 以及如何生成测试报
  • matplotlib绘图与可视化2

    文章目录 前言 一 使用pandas和seaborn绘图 1 1 折线图 1 2 柱状图 1 3 直方图和密度图 1 4 散点图或点图 1 5 分面网格和分类数据 总结 前言 matplotlib是一个相当底层的工具 你可以从其基本组件中组
  • java ioc依赖注入,Spring bean的实例化和IOC依赖注入详解

    前言 我们知道 IOC是Spring的核心 它来负责控制对象的生命周期和对象间的关系 举个例子 我们如何来找对象的呢 常见的情况是 在路上要到处去看哪个MM既漂亮身材又好 符合我们的口味 就打听她们的电话号码 制造关联想办法认识她们 然后
  • 【带头结点的单链表】

    带头结点的单链表 前言 一 带头结点的单链表结构体设计 1 带头结点的单链表 2 结构体声明 二 函数实现 1 初始化 2 申请新节点 3 头插 4 尾插 5 按位置插入 6 头删 7 尾删 8 销毁 总结 前言 单链表的概念 单链表是一种
  • CS162 操作系统HW2(使用Liunx内核链表以及多线程实现WordCounter)

    心得体会 IDE自动提示补全真的特别重要 大大提高开发效率 通过IDE自动搜索库函数API GDB调试能力要加强 使用前面提供的list h来改写wordCount程序 头文件的实现相当有技巧 将使用外部list库 多线程都用宏定义到同一份
  • Could not load dynamic library ‘libcupti.so.10.0‘; dlerror: libcupti.so.10.0...

    环境 Ubuntu 16 04 CUDA 10 0 CUDNN 7 6 5 nvcc NVIDIA R Cuda compiler driver Copyright c 2005 2018 NVIDIA Corporation Built
  • ESP32 /ESP8266在VS Code and PlatformIO上传文件系统 (SPIFFS)

    ESP32 ESP8266在VS Code and PlatformIO上传文件系统 SPIFFS 学习如何上传文件到ESP32板文件系统 SPIFFS 使用VS Code与PlatformIO IDE扩展 快速和简单 使用ESP32的文件
  • 【计算机毕业设计】课堂考勤微信小程序 基于微信小程序的课堂考勤管理系统

    毕设帮助 源码交流 技术解答 见文末 一 前言 在目前国内的高校课堂考勤中 传统的到场点名方式耗费了教师大量的时间和精力 随着课堂人数的增加 学生群体呈现多样性 这种点名考勤方式将不再适合日常使用 而且传统的点名考勤无法避免代人答到现象 极
  • 包装类这颗语法糖,其实并不甜

    历史文章推荐 你真的了解时间吗 细数ThreadLocal三大坑 内存泄露仅是小儿科 Java 8 ConcurrentHashMap源码中竟然隐藏着两个BUG ConcurrentHashMap中有十个提升性能的细节 你都知道吗 Hash