Java 中的拆箱和装箱
在 Java 中,每个基本数据类型都对应了一个包装类型,比如:int 的包装类型是 Integer,double 的包装类型是 Double…那么,基本数据类型和包装类型有什么区别呢?
大概有以下几点区别:
- 成员变量的包装类型不赋值就是 null,而基本数据类型有其默认值并且不是 null。
- 包装类型可以用于泛型,而基本数据类型不可以用于泛型。
- 基本数据类型的局部变量(在方法中声明的变量)存放在 Java 虚拟机栈中的局部变量表中,基本数据类型的成员变量(在类中声明的变量)存放在 Java 虚拟机的堆中;而包装类型属于对象类型,对象实例都存放在堆中。
- 相比于对象类型,基本数据类型占用的空间非常小。
- 两个包装类型的值可以相同,但是却不相等。
注意:基本数据类型存放在栈中是一个常见的误区!基本数据类型的局部变量(在方法中声明的变量)存放在方法栈中;基本数据类型的成员 变量(在类中声明的变量,也叫全局变量)存放在堆中。
我们主要是从基本数据类型和包装类型的数据转换中来区别拆箱和装箱:
- 拆箱:将包装类型转为基本数据类型;
- 装箱:将基本数据类型转为包装类型。
1. 包装类型可以为 null,而基本数据类型不可以
这一点区别使得包装类型可以应用于 POJO(Plain Ordinary Java Object)中,POJO 就是简单无规则的 Java 对象,只有字段及其对应的 setter 和 getter 方法:
class Person {private Integer age;private String name;public Integer getAge() {return age;}public void setAge(Integer age) {this.age = age;}public String getName() {return name;}public void setName(String name) {this.name = name;}
}
同样地,还有 DTO(Data Transfer Object)数据传输对象,泛指用于 View 层与 Service 层之间的数据传输对象;VO(View Object)视图对象,用于把向页面上展示的数据封装起来;PO(Persistant Object)持久化对象,类似于与数据库中的表映射的 Java 对象。
为什么 POJO 中的字段必须要使用包装类型呢?
《阿里巴巴 Java 开发手册》上有详细的说明:
数据库查询的结果可能是 null,如果使用基本类型的话,因为要自动拆箱,就会抛出 NullPointerException 异常(null 值拆箱异常)。
2. 包装类型可以用于泛型,而基本数据类型不可以
List<int> list = new ArrayList<>(); // 提示 Syntax error, insert "Dimensions" to complete ReferenceType
List<Integer> list = new ArrayList<>();
这里编译器会提示错误:

这是为什么呢?
因为泛型在编译时会进行类型擦除,最后只保留原始类型,而原始类型只能是 Object 类及其子类(不包括基本数据类型)。
3. 基本数据类型比包装类型更高效
作为局部变量时,基本数据类型在栈中直接存储的是具体的数值,而包装类型则存储的是堆中的引用:

显然,相比较而言,包装类型需要占用更多的内存空间,因为它不仅需要存储对象,还要存储对象的引用。也就是说,如果没有基本数据类型的话,对于数值这种经常使用到的数据来说,每次都要通过 new 一个包装类型就显得非常笨重了。
4. 两个包装类型的值可以相同,但是却不相等
/*** 拆箱和装箱* 1. 包装类型(可以应用于pojo)可以为null,基本类型不可以* 2. 包装类型可以用于泛型,基本类型不可以* 3. 基本类型比包装类型更高效* 4. 两个包装类型的值可以相同,但是不相等** @author qiaohaojie* @date 2023/3/4 18:00*/
public class PackAndUnPacking {public static void main(String[] args) {Integer a = new Integer(10);Integer b = new Integer(10);System.out.println(a == b); // falseSystem.out.println(a.equals(b)); // true}
}
两个包装类型在使用 “==” 进行判断的时候,判断的是其指向的地址是否相等,因为是两个对象,所以地址是不同的。
而 a.equals(b) 的输出结果是 true,是因为 equals() 方法内部比较的是两个 int 值是否相等:
public boolean equals(Object obj) {if (obj instanceof Integer) {return value == ((Integer)obj).intValue();}return false;
}
在 equals() 方法的源码中,((Integer)obj).intValue() 就是用来自动拆箱的。
既然有基本数据类型也有包装类型,那么在使用的时候要在它们之间进行转换:
- 把基本数据类型转换成包装类型的过程叫做装箱;
- 把包装类型转换成基本数据类型的过程叫做拆箱。
在 JDK 1.5 之前,我们要进行手动装箱和拆箱:
Integer a = new Integer(10); // 手动装箱
int b = chenmo.intValue(); // 手动拆箱
JDK 1.5 以后,为了减少开发人员的工作量,提供了自动装箱与自动拆箱的功能:
Integer a = 10; // 自动装箱
int b = a; // 自动拆箱
看一下反编译后的代码:
Integer a = Integer.valueOf(10);
int b = chenmo.intValue();
也就是说,自动装箱是通过 Integer.valueOf() 完成的;自动拆箱是通过 Integer.valueOf() 完成的。
来看一下下面的例子:
// 1. 基本类型和包装类型
int a = 100;
Integer b = 100;
System.out.println(a == b); // true// 2. 两个包装类型
Integer c = 100;
Integer d = 100;
System.out.println(c == d); // true// 3. 给包装类型重新赋值
c = 200;
d = 200;
System.out.println(c == d); // false
-
第一个结果是 true,基本类型在与包装类型进行 == 比较的时候,包装类型会自动拆箱,也就以为着两者比较的是值,值都是100,所以结果为 true.
-
第二个结果是 true,两个包装类型的被赋值为 100,这个时候会进行自动装箱,在上面的例子中我们知道,自动装箱是通过 Integer.valueOf() 方法来完成的,我们来扒一下
Integer.valueOf()方法的源码:public static Integer valueOf(int i) {if (i >= IntegerCache.low && i <= IntegerCache.high)return IntegerCache.cache[i + (-IntegerCache.low)];return new Integer(i); }这里面使用了一个静态内部类 IntegerCache,在Java中 new Integer 与 Integer.valueOf 的区别中介绍了这个静态内部类的作用及执行步骤。
其实也就需要记住一点:
当需要自动装箱时,在 -128~127 之间的数字会从 IntegerCache 中取,而不是重新创建一个对象。 -
第三个结果是 false,两个包装类型被重新赋值为 200,仍然会进行自动装箱,但是 200 不在这个缓存中,要创建两个对象,所以是 false。
相关文章:
Java 中的拆箱和装箱
在 Java 中,每个基本数据类型都对应了一个包装类型,比如:int 的包装类型是 Integer,double 的包装类型是 Double…那么,基本数据类型和包装类型有什么区别呢? 大概有以下几点区别: 成员变量的…...
从0开始自制解释器——实现多位整数的加减法计算器
上一篇我们实现了一个简单的加法计算器,并且了解了基本的词法分析、词法分析器的概念。本篇我们将要对之前实现的加法计算器进行扩展,我们为它添加以下几个功能 计算减法能自动识别并跳过空白字符不再局限于单个整数,而是能计算多位整数 提…...
(12)C#传智:File类,泛型,字典,FileStream,StreamReader,多态
内容有点多,重点:泛型、字典,流与多态。 继续深入学习内容:List、Dictionary、using语句、FileStream 一、File类的继续学心 File.ReadAllLines(string path,Encoding,encoding)指定编码读取返回行字串数组 File.WriteAllText(string…...
Dubbo的服务暴漏与服务发现源码详解
服务暴漏 如果配置需要刷新则根据配置优先级刷新服务配置 如果服务已经导出,则直接返回 是否异步导出(全局或者服务级别配置了异步,则需要异步导出服务) 服务暴漏入口DefaultModuleDeployer#exportServices private void exp…...
Python 的IDE——PyCharm
IDE介绍与安装 介绍 集成开发环境(IDE) 集成开发环境(IDE,integrated Development Environment) —— 集成开发软件需要的所有工具,一般包括以下工具: 图形用户界面 代码编辑器(支持代码补全、自动缩进) 编译器/解释器 调试器…...
01 C语言使用链表实现队列(Queue、FIFO)模块
01 C语言使用链表实现队列(Queue、FIFO)模块 作者将狼才鲸创建日期2023-03-08Gitee源码仓库地址:C语言使用链表实现队列(Queue、FIFO)模块 Linux原生的队列KFIFO一次只能操作一个队列,操作变长元素时&…...
2.2操作系统-进程管理:前趋图、前趋图与PV操作
2.1操作系统-进程管理:前趋图\前趋图与PV操作前趋图前趋图与PV操作练习前趋图与PV操作,一般出现了,分值在2~3分左右,技巧性很强。 前趋图 前趋图是为了描述一个程序的各部分间的依赖关系,或者是一个大的计算的各个子…...
凤凰游攻略
凤凰游攻略1 装备📦1.1 证件1.2 日常用品1.3 药品1.4 衣物1.5 洗漱用品2 交通🚗3 住宿🏠4 美食🍕5 拍照📷5.1 租苗族服5.1.1 单租服装5.1.2 服装化妆5.2 一条龙旅拍6 路线🗺️景点🏙️7 注意⚠️…...
Nginx 高可用方案
准备工作 10.10.4.5 10.10.4.6 VIP:10.10.4.10 两台虚拟机。安装好Nginx 安装Nginx 更新yum源文件: rpm -ivh http://nginx.org/packages/centos/7/noarch/RPMS/nginx-release-centos-7-0.el7.ngx.noarch.rpm wget -O /etc/yum.repos.d/CentOS-Ba…...
Linux基本指令
文章目录 常用Linux命令常见Linux指令 1、ls指令 语法:ls [选项][目录或文件] 功能:对于目录,该命令列出该目录下的所有子目录与文件。对于文件,将列出文件名以及其他信息。常用选项: -a 列出目录下的所有文件…...
Linux系统基础命令(二)
一、浏览和切换目录 ls命令:列出文件和目录,主要用于列出文件和目录 CentOS的终端默认是有颜色标注的。一般来说:蓝色--->目录;绿色-->可执行文件;红色--->压缩文件;浅蓝色--->链接文件&#…...
【C++】C++11——简介|列表初始|简化声明|nullptr与范围for|STL中的变化
文章目录一、C11简介二、列表初始化三、简化声明四、nullptr与范围for五、STL中一些变化一、C11简介 在2003年C标准委员会曾经提交了一份技术勘误表(简称TC1),使得C03这个名字已经取代了C98称为C11之前的最新C标准名称。不过由于TC1主要是对C98标准中的漏洞进行修复…...
Python -- 函数
文章目录1、一个简单的函数2、多参数函数3、返回值3.1、简单的返回3.2、返回列表和字典4、传入列表5、传入任意数量的实参5.1、以元组和字典的形式5.2、形参的排列顺序6、将函数储存在模块中1、一个简单的函数 函数用关键字def来定义,传参时不用指定参数类型 para&…...
Pytorch中utils.data 与torchvision简介
Pytorch中utils.data 与torchvision简介1 数据处理工具概述2 utils.data简介3 torchvision简介3.1 transforms3.2 ImageFolder1 数据处理工具概述 Pytorch涉及数据处理(数据装载、数据预处理、数据增强等)主要工具包及相互关系如下图所示,主…...
学习 Python 之 Pygame 开发魂斗罗(十)
学习 Python 之 Pygame 开发魂斗罗(十)继续编写魂斗罗1. 解决敌人不开火的问题2. 创建爆炸效果类3. 为敌人跳入河中增加爆炸效果4. 玩家击中敌人继续编写魂斗罗 在上次的博客学习 Python 之 Pygame 开发魂斗罗(九)中,…...
Keepalive+LVS群集部署
KeepaliveLVS群集部署一、Keepalive概述1、什么是Keepalive2、Keepalive工作原理3、Keepalive主要模块及作用4、Keepalived 服务重要功能(1)管理 LVS 负载均衡软件(2)支持故障自动切换(3)实现 LVS 负载调度…...
数组、指针总结【面试题】
文章目录0. 补充知识数组笔试题1. 一维数组1.1 字符数组1.1.1 sizeof1.1.2 strlen1.2 二维数组2. 指针笔试题0. 补充知识 在进入数组与指针的练习时,我们先来复习以下以下的知识点,这可以帮助我们更好的理解下面练习 数组是一组能存放相同类型的类型的元…...
七色电子标签
机种名 电子会议桌牌 型号 ESL_7color_7.3_D 外观尺寸 176.2x137.15x80mm 产品重量 268g 可视区域 163.297.92mm 外观颜色 银色 供电方式 锂电池供电2300mAh(Type-C 接口可充电) 显示技术 E-INK电子纸,双屏 像素 800x480 像…...
大数据是什么?发展前景怎么样
关于大数据的解释,比较官方的定义是指无法在一定时间范围内用常规软件工具进行捕捉、管理和处理的数据集合,是需要新处理模式才能具有更强的决策力、洞察发现力和流程优化能力的海量、高增长率和多样化的信息资产。简单来说,大数据就是结构化…...
MYSQL必知必会 | 查询相关
汇总数据 聚集函数 有时只需要汇总数据,并不需要把数据实际检索出来,所以MySql提供了专门的函数 聚集函数:运行在行组上,计算和返回单个值的函数 函数说明AVG()返回某列平均值COUNT()返回某列的行数MAX()返回某列最大值MIN()返…...
[特殊字符] 智能合约中的数据是如何在区块链中保持一致的?
🧠 智能合约中的数据是如何在区块链中保持一致的? 为什么所有区块链节点都能得出相同结果?合约调用这么复杂,状态真能保持一致吗?本篇带你从底层视角理解“状态一致性”的真相。 一、智能合约的数据存储在哪里…...
【JavaEE】-- HTTP
1. HTTP是什么? HTTP(全称为"超文本传输协议")是一种应用非常广泛的应用层协议,HTTP是基于TCP协议的一种应用层协议。 应用层协议:是计算机网络协议栈中最高层的协议,它定义了运行在不同主机上…...
R语言AI模型部署方案:精准离线运行详解
R语言AI模型部署方案:精准离线运行详解 一、项目概述 本文将构建一个完整的R语言AI部署解决方案,实现鸢尾花分类模型的训练、保存、离线部署和预测功能。核心特点: 100%离线运行能力自包含环境依赖生产级错误处理跨平台兼容性模型版本管理# 文件结构说明 Iris_AI_Deployme…...
Cesium1.95中高性能加载1500个点
一、基本方式: 图标使用.png比.svg性能要好 <template><div id"cesiumContainer"></div><div class"toolbar"><button id"resetButton">重新生成点</button><span id"countDisplay&qu…...
使用分级同态加密防御梯度泄漏
抽象 联邦学习 (FL) 支持跨分布式客户端进行协作模型训练,而无需共享原始数据,这使其成为在互联和自动驾驶汽车 (CAV) 等领域保护隐私的机器学习的一种很有前途的方法。然而,最近的研究表明&…...
3403. 从盒子中找出字典序最大的字符串 I
3403. 从盒子中找出字典序最大的字符串 I 题目链接:3403. 从盒子中找出字典序最大的字符串 I 代码如下: class Solution { public:string answerString(string word, int numFriends) {if (numFriends 1) {return word;}string res;for (int i 0;i &…...
微软PowerBI考试 PL300-在 Power BI 中清理、转换和加载数据
微软PowerBI考试 PL300-在 Power BI 中清理、转换和加载数据 Power Query 具有大量专门帮助您清理和准备数据以供分析的功能。 您将了解如何简化复杂模型、更改数据类型、重命名对象和透视数据。 您还将了解如何分析列,以便知晓哪些列包含有价值的数据,…...
Pinocchio 库详解及其在足式机器人上的应用
Pinocchio 库详解及其在足式机器人上的应用 Pinocchio (Pinocchio is not only a nose) 是一个开源的 C 库,专门用于快速计算机器人模型的正向运动学、逆向运动学、雅可比矩阵、动力学和动力学导数。它主要关注效率和准确性,并提供了一个通用的框架&…...
C#中的CLR属性、依赖属性与附加属性
CLR属性的主要特征 封装性: 隐藏字段的实现细节 提供对字段的受控访问 访问控制: 可单独设置get/set访问器的可见性 可创建只读或只写属性 计算属性: 可以在getter中执行计算逻辑 不需要直接对应一个字段 验证逻辑: 可以…...
Vite中定义@软链接
在webpack中可以直接通过符号表示src路径,但是vite中默认不可以。 如何实现: vite中提供了resolve.alias:通过别名在指向一个具体的路径 在vite.config.js中 import { join } from pathexport default defineConfig({plugins: [vue()],//…...
