[Java]关于基本数据类型与引用类型赋值时的底层分析的小结(简述)

这篇具有很好参考价值的文章主要介绍了[Java]关于基本数据类型与引用类型赋值时的底层分析的小结(简述)。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

【版权声明】未经博主同意,谢绝转载!(请尊重原创,博主保留追究权)
https://www.cnblogs.com/cnb-yuchen/p/17969159
出自【进步*于辰的博客】

目录
  • 1、关于赋值
    • 1.1 基本数据类型赋值
    • 1.2 String类型赋值
  • 2、关于String赋值
    • 2.1 情形一
    • 2.2 情形二
  • 3、关于String与char[]的比较
  • 4、不同类型引用分析
    • 4.1 int
    • 4.2 Integer
    • 4.3 int[]
    • 4.4 Integer[]
  • 5、最后

1、关于赋值

参考笔记一,P74.1。

一个小结:
所有引用都存于栈,而对象存于堆。引用所指向的可能存于栈,也可能存于方法区常量池。

1.1 基本数据类型赋值

final int a中a是常量,在int a中a是变量。
示例:int a = 10;
a是变量,10是常量(a与10都存于栈)。令a = 20,是将a的值直接由10改为20。

1.2 String类型赋值

String s = "abc"中,s是引用(也是变量,存于栈),"abc"是字符串常量,存于方法区的字符串常量池。令s = "123",是将s由指向"abc"转为指向"123",若"abc"没有其他引用指向就会被回收。

2、关于String赋值

参考笔记一,P74.2。

2.1 情形一

以下两种形式的定义的结果都相同,而底层分析略有不同。

形式一:
示例:

String s1 = "csdn", s2 = "csdn";
s2 = "2023";
sout s1;// 打印:csdn
sout s2;// 打印:2023

若方法区的字符串常量池中存在字符串常量“csdn”,则不会重新创建,直接将s1s2指向“csdn”
s2 = "2023",同样先判断是否存在"2023"。若不存在,则创建,然后将s2由指向"csdn"转为指向"2023"
因此,最终s1指向"csdn"s2指向"2023"

形式二:
示例:

String s1 = "csdn", s2 = s1;
s2 = "2023";
sout s1;// 打印:csdn
sout s2;// 打印:2023

若存在“csdn”,则不会重新创建,直接将s1指向“csdn”,然后将s2指向s1的指向。
后续步骤同上,故结果相同。

这就是为何明明s2 = s1,而s2 = "2023",结果s2改变、s1不变的原因。

2.2 情形二

示例:

String s1 = "csdn", s2 = s1 + "2023", s3 = "csdn2023";
s2.equals(s3);// 结果:true

s2 = s1 + "2023",会先获取s1的指向,是"csdn"
然后为"csdn"复制一个副本,再转为 StringBuffer,后执行append("2023"),最后调用toString(),返回“csdn2023”
此时再判断字符串常量池中是否存在字符串常量"csdn2023",若不存在,则创建,然后将s2指向csdn2023”
因此,s2s3的结果相同。

3、关于String与char[]的比较

如果大家想要了解String类的底层,看这里 → String类。
参考笔记一,P74.3。

示例:

String s1 = "csdn";
char[] arr = {'c', 's', 'd', 'n'};
s1.equals(arr);// 结果:false-------A
s1.toString().equals(arr.toString());// 结果:false-------B
// 注:第一个toString()多余,第二个toString()是在底层调用
// 我这么写是为了方便大家理解

String类重写了equals(),在底层会先调用toString(),返回值后再判断;而char[]不是具体类型,不存在重写。当然,仍会调用toString(),即示例中的A与B等效,但调用的是Object类的toString()(返回地址),故不等。

4、不同类型引用分析

参考笔记一,P36.5;笔记二,P38.1。

4.1 int

示例:

int a = 100, b = a;

a、b、100都存于栈,令a = 20,是直接将a的值改为20,b的值仍为100。

4.2 Integer

关于八位有符号二进制的表示范围,可查阅博文《关于二进制的原码、补码和反码,以及表示范围、常见位运算符和进制转换的理解与简述》的第1.2项。

Integer 类的“自动装箱”和“自动拆箱”机制是什么?

  1. 包装类都具有自动装箱和自动拆箱的功能(以代码的角度上说,就是会在底层调用某个方法)。
  2. 为包装类Integer赋值时,自动装箱是底层调用了valueOf()。这里存在一个溢出问题。因为整型常量存储于方法区的整型常量池,而整型常量池使用8位有符号二进制表示整型。8位有符号二进制的表示范围是-128 ~ 127。若整型超出此此范围,就是“溢出”。
  3. “溢出”规定:若整型在范围内,valueOf()底层创建整型常量,存储于整型常量池;否则创建 Integer 对象,存储于堆。

示例:

关于valueOf(),可参考Integer类的第4.33项。

Integer i1 = 1;
Integer i2 = new Integer(1);
Integer i3 = Integer.valueOf(1);
i1 == i2;// false
i1 == i3;// true
i2 == i3;// false

Integer i4 = 128;
Integer i5 = new Integer(128);
Integer i6 = Integer.valueOf(128);
i4 == i5;// false
i4 == i6;// false
i5 == i6;// false

底层分析:

  • i1指向整型常量1i2指向new Integer(1);由于1在表示范围内,故valueOf(1)返回整型常量1,则i3也指向整型常量1
  • 由于整型常量池仅能存储-128 ~ 127的整型常量,故i4指向new Integer(128)i5指向new Integer(128);由于128超出表示范围,故valueOf(128)返回new Integer(128)

扩展一点:

i1.equals(i2);// true
i1.equals(i3);// true
i2.equals(i3);// true

i4.equals(i5);// true
i4.equals(i6);// true
i5.equals(i6);// true

为何结果都为true

关于equals(),可参考Integer类的第4.6项。

4.3 int[]

示例:

int[] arr1 = {1, 2, 3}, arr2 = arr1;

arr1arr2以及数组内所有的值都存于栈,{1, 2, 3}存储于堆。
arr1[0] = 10,是直接将1改为10,则arr1是{10, 2, 3};arr2与arr1指向相同,故arr2也是{10, 2, 3}

4.4 Integer[]

示例:

Integer[] arr1 = {1, 2, 3}, arr2 = arr1;

int[]不同的是,Integer[]内的所有元素都不是直接的值,而是引用,并且是 Integer 引用,指向整型常量池或堆。
不过,无论指向哪里,由于arr2与arr1始终指向相同,因此,令arr1[0] = 10,则arr1是{10, 2, 3},arr2同样是{10, 2, 3}

5、最后

本文中的例子是为了方便大家理解java基本数据类型和引用类型赋值时的底层而简单举例的,不一定有实用性。

PS:
大家在看完这篇文章后肯定想吐槽:“你说了这么多,示例结果还不是与我们平时使用时的结果无差别,有什么意义。”
哈哈......结果当然相同了,因为我阐述的是底层分析。这些呢在平时工作中一般是用不到的,因此,目的是为了让大家对java基本数据类型和引用类型赋值时的底层有进一步的理解。

本文完结。文章来源地址https://www.toymoban.com/news/detail-797279.html

到了这里,关于[Java]关于基本数据类型与引用类型赋值时的底层分析的小结(简述)的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • 第五章 基本引用类型(上)——Date、RegExp

            引用值(或者对象)是某个特定引用类型的实例。在ECMAScript中,引用类型是把数据和功能组织到一起的结构,经常被人错误地称作“类”。虽然从技术上讲JavaScript是一门面向对象语言,但ECMAScript缺少传统的面向对象编程语言所具备的某些基本结构,包括类和接口

    2024年01月17日
    浏览(27)
  • 关于Swift中闭包和OC中block对局部变量基本数据类型值的捕获

    翻了很多文章,发现关于Swift闭包关于上下文变量捕获这块,都没有说的很详细,或者Swift2这样的老版本已经不适用了,问了GPT也是和自己实验的结果不一样,记录下来。 首先,回顾一下OC中的block。 block对局部变量基本数据类型的捕获,是在创建时捕获了值,并保存副本在自

    2024年02月16日
    浏览(31)
  • Java--基本数据类型

    在Java中,一共有8种基本类型,其中有4种整形,2种浮点类型,1种字符类型char(用于表示Unicode编码的代码单元)和一种表示真值bloolean类型。 1、整形 整形用于表示没有小数部分的数值,允许是负数。Java提供了4种整形。 在通常情况下,int 类型最常用。但如果想要表示整个地

    2024年02月11日
    浏览(35)
  • Java基本数据类型

    Java是一种强类型语言,它有八种基本数据类型,分为三类:整型、浮点型和字符型。如何选择适当的数据类型是一个程序员需要考虑的重要问题。在本文中,我们将详细介绍Java的基本数据类型,以及它们在程序中的应用。 整型数据类型表示整数。Java有四种整型数据类型,分

    2023年04月09日
    浏览(32)
  • Java基础篇_1.2——保留关键字、基本数据类型、基本数据类型之间的转换

    目录 一、保留 二、Java的基本数据类型 三、引用数据类型  四、基本数据类型间的转换  隐含强制类型转换 数值之间的合法转换 Java该语言是用 Unicode 字符集编写的。 Java是预先定义的具有特别意义的标识符,也被称为Java保留字,Java不能用作变量名、方

    2024年02月14日
    浏览(35)
  • 【Java SE】基本数据类型

    大家好!我是保护小周ღ,本期为大家带来的是 Java的基本数据类型,内容会与C语言的基本数据类型进行基本的比较,数据类型提示,整型提升,以及简单了解 String 类型 ,进一步感受Java 的安全性。 C语言混不下去了,面向对象的编程太爽了~ 目录 一、常量 二、基本数据类

    2024年02月11日
    浏览(24)
  • java基本数据类型取值范围

    在JAVA中一共有八种基本数据类型,他们分别是  byte、short、int、long、float、double、char、boolean  整型  其中byte、short、int、long都是表示整数的,只不过他们的取值范围不一样  byte的取值范围为-128~127,占用1个字节(-2的7次方到2的7次方-1)  short的取值范围为-32768~32767,占用

    2024年02月12日
    浏览(22)
  • Java 变量与基本数据类型

    2024年02月09日
    浏览(26)
  • 开心档之Java 基本数据类型

    目录 Java 基本数据类型 内置数据类型 实例 实例 类型默认值 实例 引用类型 Java 常量 自动类型转换 自动类型转换 实例 强制类型转换 实例 隐含强制类型转换   变量就是申请内存来存储值。也就是说,当创建变量的时候,需要在内存中申请空间。 内存管理系统根据变量的类

    2024年02月05日
    浏览(24)
  • Advanced .Net Debugging 4:基本调试任务(对象检查:内存、值类型、引用类型、数组和异常的转储)

    一、介绍 这是我的《 Advanced .Net Debugging 》这个系列的第四篇文章。今天这篇文章的标题虽然叫做“基本调试任务”,但是这章的内容还是挺多的。由于内容太多,故原书的第三章内容我分两篇文章来写。上一篇我们了解了一些调试技巧,比如:单步调试、下断点、过程调试

    2024年03月09日
    浏览(30)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包