Java 中为什么要设计包装类
尽人事,听天命。博主东南大学硕士在读,热爱健身和篮球,乐于分享技术相关的所见所得,关注公众号 @ 飞天小牛肉,第一时间获取文章更新,成长的路上我们一起进步
本文已收录于 「CS-Wiki」Gitee 官方推荐项目,现已累计 1.5k+ star,致力打造完善的后端知识体系,在技术的路上少走弯路,欢迎各位小伙伴前来交流学习
如果各位小伙伴春招秋招没有拿得出手的项目的话,可以参考我写的一个项目「开源社区系统 Echo」Gitee 官方推荐项目,目前已累计 400+ star,基于 SpringBoot + MyBatis + MySQL + Redis + Kafka + Elasticsearch + Spring Security + ... 并提供详细的开发文档和配套教程。公众号后台回复 Echo 可以获取配套教程,目前尚在更新中
最近文章更新频率慢了,因为最近在准备暑期实习,之前寻思着一边复习一边写文章,两全其美。后来发现一篇读起来比较舒服的文章写出来加上配图得花上四五个小时甚至更多,但这个知识点我可能半个小时就能复习完了,春招在即,时间比较紧迫,所以最近文章可能改为一周一更或者一周两更,希望各位理解。另外,有和我一样在准备暑期实习的小伙伴可以联系我互相交流
全文脉络思维导图如下:
1. 为什么需要包装类
在 Java 中,万物皆对象,所有的操作都要求用对象的形式进行描述。但是 Java 中除了对象(引用类型)还有八大基本类型,它们不是对象。那么,为了把基本类型转换成对象,最简单的做法就是将基本类型作为一个类的属性保存起来,也就是把基本数据类型包装一下,这也就是包装类的由来。
这样,我们先自己实现一个简单的包装类,以包装基本类型 int
为例:
// 包装类 MyInt
public class MyInt {
private int number; // 基本数据类型
public Int (int number){ // 构造函数,传入基本数据类型
this.number = number;
}
public int intValue(){ // 取得包装类中的数据
return this.number;
}
}
测试一下这个包装类:
public static void main(String[] args) {
MyInt temp = new Int(100); // 100 是基本数据类型, 将基本数据类型包装后成为对象
int result = temp.intValue(); // 从对象中取得基本数据类型
System.out.println(result);
}
当然,我们自己实现的这个包装类非常简单,Java 给我们提供了更完善的内置包装类:
基本类型 | 对应的包装类(位于 java.lang 包中) |
---|---|
byte | Byte |
short | Short |
int | Integer |
long | Long |
float | Float |
double | Double |
char | Character |
boolean | Boolean |
前 6 个类派生于公共的超类 Number
,而 Character
和 Boolean
是 Object
的直接子类。
来看看包装类的声明,以 Integer
为例:
被 final
修饰,也就是说 Java 内置的包装类是无法被继承的。
2. 装箱与拆箱
OK,现在我们已经知道了,存在基本数据类型与其对应的包装类,那么,他们之间互相的转换操作就称为装箱与拆箱:
- 装箱:将基本数据类型转换成包装类(每个包装类的构造方法都可以接收各自数据类型的变量)
- 拆箱:从包装类之中取出被包装的基本类型数据(使用包装类的 xxxValue 方法)
下面以 Integer
为例,我们来看看 Java 内置的包装类是如何进行拆装箱的:
Integer obj = new Integer(10); // 自动装箱
int temp = obj.intValue(); // 自动拆箱
可以看出,和上面我们自己写的包装类使用方式基本一样,事实上,Integer
中的这两个方法其底层实现和我们上述写的代码也是差不多的。
不知道各位发现没,value
被声明为 final
了,也就是说一旦构造了包装器,就不允许更改包装在其中的值。
另外,需要注意的是,这种形式的代码是 JDK 1.5 以前的!!!JDK 1.5 之后,Java 设计者为了方便开发提供了自动装箱与自动拆箱的机制,并且可以直接利用包装类的对象进行数学计算。
还是以 Integer
为例我们来看看自动拆装箱的过程:
Integer obj = 10; // 自动装箱. 基本数据类型 int -> 包装类 Integer
int temp = obj; // 自动拆箱. Integer -> int
obj ++; // 直接利用包装类的对象进行数学计算
System.out.println(temp * obj);
看见没有,基本数据类型到包装类的转换,不需要像上面一样使用构造函数,直接 =
就完事儿;同样的,包装类到基本数据类型的转换,也不需要我们手动调用包装类的 xxxValue 方法了,直接 =
就能完成拆箱。这也是将它们称之为自动的原因。
我们来看看这段代码反编译后的文件,底层到底是什么原理:
Integer obj = Integer.valueOf(10);
int temp = obj.intValue();
可以看见,自动装箱的底层原理是调用了包装类的 valueOf
方法,而自动拆箱的底层调用了包装类的 intValue()
方法。
3. 不简单的 Integer.valueOf
我们上面已经看过了用于自动拆箱的 intValue
方法的源码,非常简单。接下来咱来看看用于自动装箱的 valueOf
,其他包装类倒没什么好说的,不过 Integer
中的这个方法还是有点东西的:
IntegerCache
又是啥,点进去看看:
IntegerCache
是 Integer
类中的静态内部类,综合这两段代码,我们大概也能知道,IntegerCache
其实就是个缓存,其中定义了一个缓冲区 cache
,用于存储 Integer
类型的数据,缓存区间是 [-128, 127]。
回到 valueOf
的源码:它首先会判断 int 类型的实参 i 是否在可缓存区间内,如果在,就直接从缓存 IntegerCache
中获取对应的 Integer
对象;如果不在缓存区间内,则会 new 一个新的 Integer
对象。
结合这个特性,我们来看一个题目,两种类似的代码逻辑,但是却得到完全相反的结果。:
public static void main(String args[]) {
Integer a1 = 127;
Integer a2 = 127;
System.out.println(a1 == a2); // true
Integer b1 = 128;
Integer b2 = 128;
System.out.println(b1 == b2); // false
}
我们知道,==
拥有两种应用场景:
- 对于引用类型来说,判断的是内存地址是否相等
- 对于基本类型来说,判断的是值是否相等
从 a1 开始看,由于其值在 InterCache
的缓存区间内,所以这个 Integer
对象会被存入缓存。而在创建 a2 的时候,由于其值和 a1 相等,所以直接从缓存中取出值为 127 的 Integer
对象给 a2 使用,也就是说,a1 和 a2 这两个 Integer
的对象引用都指向同一个地址。
对于 b1 和 b2 来说,由于 128 不在 IntegerCache
的缓存区间内,那就只能自己老老实实开辟空间了,所以 b1 和 b2 指向不同的内存地址。
很显然,由于 InterCache
缓存机制的存在,可能会让我们在编程的时候出现困惑,因此最好使用 .equals
方法来比较 Integer
值是否相等。Integer
重写了 .equals
方法:
当然,其他包装类虽然没有缓存机制,但是也都重载了 .equals
方法,用于根据值来判断是否相等。因此,得出结论,使用 equals
方法来比较两个包装类对象的值。
4. Object 类可以接收所有数据类型
综上,有了自动拆装箱机制,基本数据类型可以自动的被转为包装类,而 Object
是所有类的父类,也就是说,Object
可以接收所有的数据类型了(引用类型、基本类型)!!!
不信你可以试试,直接用 Object
类接收一个基本数据类型 int
,完全是可以的。
Object obj = 10;
int temp = (Integer) obj;
解释一下上面这段代码发生了什么,下面这张图很重要,大家仔细看:
5. 包装类在集合中的广泛使用
其实包装类最常见的使用就是在集合中,因为集合不允许存储基本类型的数据,只能存储引用类型的数据。那如果我们想要存储 1、2、3 这样的基本类型数据怎么办?举个例子,我们可以如下声明一个 Integer
对象的数组列表:
ArrayList<Integer> list = new ArrayList<>();
往这个列表中添加 int
型数据:
list.add(3);
上面这个调用在底层将会发生自动装箱操作:
list.add (Integer.valueOf(3));
基本数据类型 int
会被转换成 Integer
对象存入集合中。
我们再来从这个集合中根据某个下标 i 获取对应的 Integer
对象,并用基本数据类型 int
接收:
int n = list.get(i);
上面这个调用在底层将会发生自动拆箱操作:
int n = list.get(i).intValue();
6. 数据类型转换
另外,除了在集合中的广泛应用,包装类还包含一个重要功能,那就是提供将String型数据变为基本数据类型的方法,使用几个代表的类做说明:
Integer
:
Double
:
Boolean
:
这些方法均被 static
标识,也就是说它们被各自对应的所有对象共同维护,直接通过类名访问该方法。举个例子:
String str = "10";
int temp = Integer.parseInt(str);// String -> int
System.out.println(temp * 2); // 20
需要特别注意的是:Character
类里面并不存在字符串变为字符的方法,因为 String
类中已经有一个 charAt()
的方法可以根据索引取出字符内容。
关注公众号 | 飞天小牛肉,即时获取更新
- 博主东南大学硕士在读,利用课余时间运营一个公众号『 飞天小牛肉 』,2020/12/29 日开通,专注分享计算机基础(数据结构 + 算法 + 计算机网络 + 数据库 + 操作系统 + Linux)、Java 基础和面试指南的相关原创技术好文。本公众号的目的就是让大家可以快速掌握重点知识,有的放矢。希望大家多多支持哦,和小牛肉一起成长
- 并推荐个人维护的开源教程类项目: CS-Wiki(Gitee 推荐项目,现已累计 1.5k+ star), 致力打造完善的后端知识体系,在技术的路上少走弯路,欢迎各位小伙伴前来交流学习 ~
- 如果各位小伙伴春招秋招没有拿得出手的项目的话,可以参考我写的一个项目「开源社区系统 Echo」Gitee 官方推荐项目,目前已累计 400+ star,基于 SpringBoot + MyBatis + MySQL + Redis + Kafka + Elasticsearch + Spring Security + ... 并提供详细的开发文档和配套教程。公众号后台回复 Echo 可以获取配套教程,目前尚在更新中。
Java 中为什么要设计包装类的更多相关文章
- Java中基本数据类型和包装类
参考:深入剖析Java中的装箱和拆箱; Java中基本数据类型和包装类互转中 缓冲机制的使用; java学习笔记:装箱和拆箱,包装器和缓冲池 Java 各 类型数据在内存中分配情况详解 一 java内 ...
- Java中特殊的类——包装类
Java中特殊的类--包装类 包装类就是将基本数据类型封装在类中. 1.包装类 (1)自定义包装类 将基本数据类型包装成一个类对象的本质就是使用Object进行接收处理. 此时IntDemo类就是in ...
- java中HashMap的设计精妙在哪?
摘要:本文结合图解和问题,教你一次性搞定HashMap 本文分享自华为云社区<java中HashMap的设计精妙在哪?用图解和几个问题教你一次性搞定HashMap>,作者:breakDaw ...
- java中的String设计原理
首先,必须强调一点:String Pool不是在堆区,也不是在栈区,而是存在于方法区(Method Area) 解析: String Pool是常量池(Constant Pool)中的一块. 我们知 ...
- Java中ThreadLocal的设计与使用
早在Java 1.2推出之时,Java平台中就引入了一个新的支持:java.lang.ThreadLocal,给我们在编写多线程程序时提供了一种新的选择.使用这个工具类可以很简洁地编写出优美的多线程程 ...
- Java中异常处理和设计
在程序设计中,进行异常处理是非常关键和重要的一部分.一个程序的异常处理框架的好坏直接影响到整个项目的代码质量以及后期维护成本和难度.试想一下,如果一个项目从头到尾没有考虑过异常处理,当程序出错从哪里寻 ...
- Java中常用类(包装类扩展知识)
Java常用类有哪些? 八大基本数据类型的包装类 包装类均位于java.lang包中,包装类和基本数据类型的对应关系如下表: 基本数据类型 包装类 byte Byte boolean Boolean ...
- Java中的基本数据类型包装类
在 java 中为什么会有基本数据类型的包装类? ①:基本数据类型之间的相互转换不是都可以制动转换的,而你强制转换又会出问题,比如String类型的转换为int类型的,那么jdk为了方便用户就提供了相 ...
- Java中的基本类型和包装类
Java中基本数据类型与包装类型有 基本类型 包装器类型 boolean Boolean char Character int Integer byte Byte short Shor ...
随机推荐
- A - 敌兵布阵 ——B - I Hate It——C - A Simple Problem with Integers(线段树)
C国的死对头A国这段时间正在进行军事演习,所以C国间谍头子Derek和他手下Tidy又开始忙乎了.A国在海岸线沿直线布置了N个工兵营地,Derek和Tidy的任务就是要监视这些工兵营地的活动情况.由于 ...
- 对模拟器虚假设备识别能力提升15%!每日清理大师App集成系统完整性检测
前言 每日清理大师是一款智能便捷的手机清理软件,可快速清理无用缓存.垃圾文件和应用残留,还可深度清理如社交软件中的无用缓存等,有效解决手机卡顿.耗电快.内存不足等问题.每日清理大师App在结合了系统完 ...
- ++i和i++的区别
它们两个的数值变化的区别,我这里就不多说了 这里主要说明两者在效率上的区别 (1)首先如果是自带的数据类型,比如int型,++i和i++,编译器的实现方式是相同的,两者并没有效率上的区别,虽然也有副本 ...
- 关于FFT的一些理解,以及如何手工计算FFT加深理解和验证正确性
总结缺少逻辑性和系统性,主要便于自己理解和记忆 关于一维FFT的 于是复系数Cn是图像傅里叶变换的yn列向量 于是我们看到最后引入,Cn这个复系数的模来表征频率波的振幅记为Sn(即简谐波叠加的数量 然 ...
- kmp--考研写法
首先是模式串匹配: #include<iostream> #include<stdlib.h> using namespace std; #define maxn 100000 ...
- 51nod1089 最长回文子串 manacher算法
0. 问题定义 最长回文子串问题:给定一个字符串,求它的最长回文子串长度. 如果一个字符串正着读和反着读是一样的,那它就是回文串.下面是一些回文串的实例: 12321 a aba abba aaaa ...
- 内存耗尽后Redis会发生什么
前言 作为一台服务器来说,内存并不是无限的,所以总会存在内存耗尽的情况,那么当 Redis 服务器的内存耗尽后,如果继续执行请求命令,Redis 会如何处理呢? 内存回收 使用Redis 服务时,很多 ...
- 关于Google Chrome的使用小技巧!
1 1 http://www.runoob.com/w3cnote/chrome-skip.html Chrome是Google公司开发的一个现代化的网页浏览器,作为三大浏览器之一 它搭载了被称为V8 ...
- RESTful 架构 && RESTful API
RESTful 架构 && RESTful API REpresentational State Transfer (REST) 具象状态传输https://en.wikipedia. ...
- 2018 free pdf ebooks
2018 free pdf ebooks https://gist.github.com/xgqfrms-GitHub/f606efb0d4bce884c873518647e79f2f https:/ ...