热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

Java中包装类的设计原因以及操作方法

本文主要介绍了Java中设计包装类的原因以及操作方法。在Java中,除了对象类型,还有八大基本类型,为了将基本类型转换成对象,Java引入了包装类。文章通过介绍包装类的定义和实现,解答了为什么需要包装类的问题,并提供了简单易用的操作方法。通过本文的学习,读者可以更好地理解和应用Java中的包装类。

这篇文章主要介绍“Java中设计了包装类的原因有哪些”,在日常操作中,相信很多人在Java中设计了包装类的原因有哪些问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”Java中设计了包装类的原因有哪些”的疑惑有所帮助!接下来,请跟着小编一起来学习吧!

全文脉络思维导图如下:

Java中设计了包装类的原因有哪些

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 包中)
byteByte
shortShort
intInteger
longLong
floatFloat
doubleDouble
charCharacter
booleanBoolean

前 6 个类派生于公共的超类 Number,而 Character 和 Boolean 是 Object 的直接子类。

来看看包装类的声明,以 Integer 为例:

Java中设计了包装类的原因有哪些

被 final 修饰,也就是说 Java 内置的「包装类是无法被继承的」。

2. 装箱与拆箱

OK,现在我们已经知道了,存在基本数据类型与其对应的包装类,那么,他们之间互相的转换操作就称为装箱与拆箱:

  • 装箱:将基本数据类型转换成包装类(每个包装类的构造方法都可以接收各自数据类型的变量)

  • 拆箱:从包装类之中取出被包装的基本类型数据(使用包装类的 xxxValue 方法)

下面以 Integer 为例,我们来看看 Java 内置的包装类是如何进行拆装箱的:

Integer obj = new Integer(10);  // 自动装箱 int temp = obj.intValue();   // 自动拆箱

可以看出,和上面我们自己写的包装类使用方式基本一样,事实上,Integer 中的这两个方法其底层实现和我们上述写的代码也是差不多的。

Java中设计了包装类的原因有哪些

不知道各位发现没,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 方法了,直接 = 就能完成拆箱。这也是将它们称之为自动的原因。

Java中设计了包装类的原因有哪些

我们来看看这段代码反编译后的文件,底层到底是什么原理:

Integer obj = Integer.valueOf(10); int temp = obj.intValue();

可以看见,自动装箱的底层原理是调用了包装类的 valueOf 方法,而自动拆箱的底层调用了包装类的 intValue() 方法。

3. 不简单的 Integer.valueOf

我们上面已经看过了用于自动拆箱的 intValue 方法的源码,非常简单。接下来咱来看看用于自动装箱的  valueOf,其他包装类倒没什么好说的,不过 Integer 中的这个方法还是有点东西的:

Java中设计了包装类的原因有哪些

IntegerCache 又是啥,点进去看看:

Java中设计了包装类的原因有哪些

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  的对象引用都指向同一个地址。

Java中设计了包装类的原因有哪些

对于 b1 和 b2 来说,由于 128 不在 IntegerCache 的缓存区间内,那就只能自己老老实实开辟空间了,所以 b1 和 b2  指向不同的内存地址。

很显然,由于 InterCache 缓存机制的存在,可能会让我们在编程的时候出现困惑,因此最好使用 .equals 方法来比较 Integer  值是否相等。Integer 重写了 .equals 方法:

Java中设计了包装类的原因有哪些

当然,其他包装类虽然没有缓存机制,但是也都重载了 .equals 方法,用于根据值来判断是否相等。因此,得出结论,「使用 equals  方法来比较两个包装类对象的值」。

4. Object 类可以接收所有数据类型

综上,有了自动拆装箱机制,基本数据类型可以自动的被转为包装类,而 Object  是所有类的父类,也就是说,「Object 可以接收所有的数据类型了」(引用类型、基本类型)!!!

不信你可以试试,直接用 Object 类接收一个基本数据类型 int,完全是可以的。

Object obj = 10; int temp = (Integer) obj;

解释一下上面这段代码发生了什么,下面这张图很重要,大家仔细看:

Java中设计了包装类的原因有哪些

5. 包装类在集合中的广泛使用

其实包装类最常见的使用就是在集合中,因为集合不允许存储基本类型的数据,只能存储引用类型的数据。那如果我们想要存储 1、2、3  这样的基本类型数据怎么办?举个例子,我们可以如下声明一个 Integer对象的数组列表:

ArrayList list = new ArrayList<>();

往这个列表中添加 int 型数据:

list.add(3);

上面这个调用在底层将会发生自动装箱操作:

int n = list.get(i);

基本数据类型 int 会被转换成 Integer 对象存入集合中。

我们再来从这个集合中根据某个下标 i 获取对应的 Integer 对象,并用基本数据类型 int 接收:

int n = list.get(i);

上面这个调用在底层将会发生自动拆箱操作:

int n = list.get(i).intValue();

6.  数据类型转换

另外,除了在集合中的广泛应用,包装类还包含一个重要功能,那就是提供将String型数据变为基本数据类型的方法,使用几个代表的类做说明:

Integer:

Java中设计了包装类的原因有哪些

Double:

Java中设计了包装类的原因有哪些

Boolean:

Java中设计了包装类的原因有哪些

这些方法均被 static 标识,也就是说它们被各自对应的所有对象共同维护,直接通过类名访问该方法。举个例子:

String str = "10";int temp = Integer.parseInt(str);// String ->  intSystem.out.println(temp * 2); // 20

需要特别注意的是:Character 类里面并不存在字符串变为字符的方法,因为 String 类中已经有一个  charAt()的方法可以根据索引取出字符内容。

Java中设计了包装类的原因有哪些

到此,关于“Java中设计了包装类的原因有哪些”的学习就结束了,希望能够解决大家的疑惑。理论与实践的搭配能更好的帮助大家学习,快去试试吧!若想继续学习更多相关知识,请继续关注编程笔记网站,小编会继续努力为大家带来更多实用的文章!


推荐阅读
  • 本指南从零开始介绍Scala编程语言的基础知识,重点讲解了Scala解释器REPL(读取-求值-打印-循环)的使用方法。REPL是Scala开发中的重要工具,能够帮助初学者快速理解和实践Scala的基本语法和特性。通过详细的示例和练习,读者将能够熟练掌握Scala的基础概念和编程技巧。 ... [详细]
  • 本文总结了JavaScript的核心知识点和实用技巧,涵盖了变量声明、DOM操作、事件处理等重要方面。例如,通过`event.srcElement`获取触发事件的元素,并使用`alert`显示其HTML结构;利用`innerText`和`innerHTML`属性分别设置和获取文本内容及HTML内容。此外,还介绍了如何在表单中动态生成和操作``元素,以便更好地处理用户输入。这些技巧对于提升前端开发效率和代码质量具有重要意义。 ... [详细]
  • 本文探讨了 Java 中 Pair 类的历史与现状。虽然 Java 标准库中没有内置的 Pair 类,但社区和第三方库提供了多种实现方式,如 Apache Commons 的 Pair 类和 JavaFX 的 javafx.util.Pair 类。这些实现为需要处理成对数据的开发者提供了便利。此外,文章还讨论了为何标准库未包含 Pair 类的原因,以及在现代 Java 开发中使用 Pair 类的最佳实践。 ... [详细]
  • Netty框架中运用Protobuf实现高效通信协议
    在Netty框架中,通过引入Protobuf来实现高效的通信协议。为了使用Protobuf,需要先准备好环境,包括下载并安装Protobuf的代码生成器`protoc`以及相应的源码包。具体资源可从官方下载页面获取,确保版本兼容性以充分发挥其性能优势。此外,配置好开发环境后,可以通过定义`.proto`文件来自动生成Java类,从而简化数据序列化和反序列化的操作,提高通信效率。 ... [详细]
  • HBase Java API 进阶:过滤器详解与应用实例
    本文详细探讨了HBase 1.2.6版本中Java API的高级应用,重点介绍了过滤器的使用方法和实际案例。首先,文章对几种常见的HBase过滤器进行了概述,包括列前缀过滤器(ColumnPrefixFilter)和时间戳过滤器(TimestampsFilter)。此外,还详细讲解了分页过滤器(PageFilter)的实现原理及其在大数据查询中的应用场景。通过具体的代码示例,读者可以更好地理解和掌握这些过滤器的使用技巧,从而提高数据处理的效率和灵活性。 ... [详细]
  • 在MFC框架中,存在多个全局函数,用于在不同对象间获取信息或创建新对象。其中,`afxGetApp`函数尤为关键,它能够帮助开发者轻松获取当前应用程序的实例指针。本文将详细解析`afxGetApp`函数的内部机制及其在MFC应用程序中的具体应用场景,探讨其在提升代码可维护性和灵活性方面的优势。此外,还将介绍其他常用全局函数如`AfxWinInit()`和`AfxBeginThread()`的功能和使用方法,为开发者提供全面的参考。 ... [详细]
  • 基址获取与驱动开发:内核中提取ntoskrnl模块的基地址方法解析
    基址获取与驱动开发:内核中提取ntoskrnl模块的基地址方法解析 ... [详细]
  • 在托管C++中开发应用程序时,遇到了如何声明和操作字符串数组的问题。本文详细探讨了字符串数组在托管C++中的应用与实现方法,包括声明、初始化、遍历和常见操作技巧,为开发者提供了实用的参考和指导。 ... [详细]
  • 在Java编程中,`String`对象既可以用作对象,也可以用作基本类型。本文深入解析了`String`对象中`equals`方法与`==`运算符的区别及其应用场景。`equals`方法用于比较两个字符串的内容是否相同,而`==`运算符则用于比较两个字符串对象的引用是否相同。通过具体示例和代码片段,文章详细阐述了这两种比较方式的内在机制和适用场景,帮助开发者更好地理解和使用`String`对象的比较操作。 ... [详细]
  • Python内置模块详解:正则表达式re模块的应用与解析
    正则表达式是一种强大的文本处理工具,通过特定的字符序列来定义搜索模式。本文详细介绍了Python内置的`re`模块,探讨了其在字符串匹配、验证和提取中的应用。例如,可以通过正则表达式验证电子邮件地址、电话号码、QQ号、密码、URL和IP地址等。此外,文章还深入解析了`re`模块的各种函数和方法,提供了丰富的示例代码,帮助读者更好地理解和使用这一工具。 ... [详细]
  • Squaretest:自动生成功能测试代码的高效插件
    本文将介绍一款名为Squaretest的高效插件,该工具能够自动生成功能测试代码。使用这款插件的主要原因是公司近期加强了代码质量的管控,对各项目进行了严格的单元测试评估。Squaretest不仅提高了测试代码的生成效率,还显著提升了代码的质量和可靠性。 ... [详细]
  • 第六章:枚举类型与switch结构的应用分析
    第六章深入探讨了枚举类型与 `switch` 结构在编程中的应用。枚举类型(`enum`)是一种将一组相关常量组织在一起的数据类型,广泛存在于多种编程语言中。例如,在 Cocoa 框架中,处理文本对齐时常用 `NSTextAlignment` 枚举来表示不同的对齐方式。通过结合 `switch` 结构,可以更清晰、高效地实现基于枚举值的逻辑分支,提高代码的可读性和维护性。 ... [详细]
  • iOS 设备唯一标识获取的高效解决方案与实践
    在iOS 7中,苹果公司再次禁止了对MAC地址的访问,使得开发者无法直接获取设备的物理地址。为了在开发过程中实现设备的唯一标识,苹果推荐使用Keychain服务来存储和管理唯一的标识符。此外,还可以结合其他技术手段,如UUID和广告标识符(IDFA),以确保设备的唯一性和安全性。这些方法不仅能够满足应用的需求,还能保护用户的隐私。 ... [详细]
  • 无论是计算机专业学生还是非计算机专业的学习者,在掌握C语言的过程中可能会遇到诸多挑战,不清楚从何入手。为此,本文系统地梳理了2019年福建省C语言的核心知识点,并结合最新的技术进展进行了详细总结,旨在为初学者提供全面的学习指导。文章不仅涵盖了基础语法和数据结构,还深入探讨了指针、内存管理和算法优化等高级主题,帮助读者快速提升编程能力。 ... [详细]
  • 针对NOJ1102黑白图像问题,本文采用深度优先搜索算法进行详细分析与实现。该问题要求在给定的时间限制(普通Java为1000-3000毫秒)和内存限制(65536KByte)内,处理一个n×n的黑白图像。通过对图像的逐像素遍历,利用深度优先搜索算法有效地识别并标记相连的黑色区域,从而实现图像的高效处理。实验结果显示,该方法在多种测试用例中均能稳定达到预期效果,具有较高的准确性和效率。 ... [详细]
author-avatar
鸡__腿孜然小朋友
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有