Java基础--基本数据类型与引用数据类型在内存结构上的区别

简介: Java基础--基本数据类型与引用数据类型在内存结构上的区别

基本数据类型与引用数据类型在内存结构上的区别

经过上一节的学习,咱们对数组有了大致的了解,包括如何定义使用,不清楚的小伙伴戳这里哈,非常详尽!!!

一/ 引入

【案例一】看一下下面的输出结果是多少。

public class TestArray2{
    public static void main(String[] args){
        int a=10;
        int b=a;
        b=100;
        System.out.println(a);
    }
}

毫无疑问,答案是10。

看一下运行结果吧:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-mJL3Y4Pp-1658541704882)(D:\Typora图片\image-20220723083519820.png)]

【案例二】看一下下面的输出结果是多少。

public class TestArray2{
    public static void main(String[] args){
        int[] x={10,20,30};
        int[] y=x;
        y[0]=100;
        System.out.println(x[0]);
    }
}

看一下结果:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-hFgS0tB6-1658541704886)(D:\Typora图片\image-20220723084023106.png)]

这个结果,很多人就会有疑惑了(疑惑是10,还是100)。

数组之所以难,不是在它的使用。而是它底层的原理,你到底有没有弄明白。

别着急,往后看,你会越来越通透。

二/ 内存存储

【案例一】

public class TestArray2{
    public static void main(String[] args){
        int a=10;
        int b=a;
        b=100;
        System.out.println(a);
    }
}

上面的案例一,内存上是如何执行的?

为了方便理解,画图解释,如下:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-UifMlIe5-1658541704886)(D:\Typora图片\image-20220723085759058.png)]

【案例二】

public class TestArray2{
    public static void main(String[] args){
        int[] x={10,20,30};
        int[] y=x;
        y[0]=100;
        System.out.println(x[0]);
    }
}

上面的案例二,内存上又是如何执行的?

之前我们说过变量,变量在内存中:

1.是栈内存中的小容器,类型定义了只能存储这种东西。

2.容器中只能存一份。


int[] x={10,20,30};

现在可以看到,x是一个数组,三个数是不能像变量那样存入一个空间的。

先来看一下int[] x={10,20,30};的标准写法:int[] x=new int[]{10,20,30};

new是新建的意思。

:star:==以后只要见到new关键字,相当于在堆内存空间中申请开辟了一块新的空间。==

既然新建了一块空间,那这块空间长啥样呢?

就是new关键字后面的内容,即开辟一块int[]数组类型的空间。

那数组类型的空间又长啥样?

:star:==数组在堆内存中的空间形态,是一串连续的地址。==

每一个小格子存的是int类型的数据。

整个开辟空间并存完东西的过程,叫做 数组的初始化

如下图:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-LfzcIvyC-1658541704886)(D:\Typora图片\image-20220723092015246.png)]

:question: 既然创建完成了,那么如何把10,20,30存入x空间里面呢?

既然在堆内存中是一串连续的地址。

那么,就把第一个地址交给x处理。

:star: ==x里面存的其实是地址(门牌号)。==

我们可以通过x里面存放的地址,找到数据在堆内存里面存储的位置。

第一个位置找到了,后面的是连续的,就都找到了。

现在暂时理解为是地址,其实是一个HashCode,以后再说这个。

如下图:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-sKveXRdz-1658541704887)(D:\Typora图片\image-20220723092747594.png)]


int[] y=x;

第二步,继续创建一个y空间,名字叫y,存储类型为int[]数组类型。

y空间的值是x给的。

x里面存的是一个地址

那么就是,==x将自己空间的地址,复制了一份,交给了y。==

如下图:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-EODeruvx-1658541704887)(D:\Typora图片\image-20220723093325417.png)]

既然y里面存的是地址,那么就可以通过这个地址访问到堆内存中的数据


y[0]=100;

再来看最后一步,这行代码的含义就是:

通过y里面存储的地址,找到堆内存中的数据(即10所在的位置)。

然后在常量区内复制了一份100,交给了堆内存中第一个位置。(将第一个数据,改成了100)

这里是y[0],就是第一个位置,即10。所以覆盖的就是10啦。

如下图:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-qnU3iYJr-1658541704888)(D:\Typora图片\image-20220723094149809.png)]


System.out.println(x[0]);

虽然输出的是x[0],但是x里面的地址指向的堆内存的数据,发生了变化。

第一个数据已经变成了100,所以最后输出,就是100啦。

:red_car:为了方便大家理解,在这里再举个小例子。可以对照着看我上面画的图哈。

小王(x)和小明(y)两个人,都有同一个仓库的钥匙(地址)。

现在小明(y)将仓库里面的西瓜拿走了,装入了菠萝。

过一会,小王(x)再次打开仓库,里面存放的就是菠萝啦。

三/ 总结

(1)基本类型变量空间存储的是,传递的是,如【案例一】。

一个改变,另一个不变。

(2)引用类型变量空间存储的是地址(引用),传递的是地址(引用),指向同一个空间。

空间内数据改动了,那就都会改动,如【案例二】。

一个改变,另一个跟着改变。

(3)所有的变量空间都存储在栈内存里。(简单来说,等号左边的东西在栈内存里面)

变量空间可以存储基本数据类型,也可以存储引用数据类型。

如果变量空间存储的是基本数据类型,存储的是,一个变量改变,另一个不会跟着改变。

(每一个变量里面存储的是一个固定的值,别人变化和我无关)。

如果变量空间存储的是引用数据类型,存储的是地址(引用),一个变量地址对应的值改变,另一个跟着改变。

相关文章
|
3月前
|
安全 Java 应用服务中间件
Spring Boot + Java 21:内存减少 60%,启动速度提高 30% — 零代码
通过调整三个JVM和Spring Boot配置开关,无需重写代码即可显著优化Java应用性能:内存减少60%,启动速度提升30%。适用于所有在JVM上运行API的生产团队,低成本实现高效能。
320 3
|
4月前
|
存储 缓存 Java
Java数组全解析:一维、多维与内存模型
本文深入解析Java数组的内存布局与操作技巧,涵盖一维及多维数组的声明、初始化、内存模型,以及数组常见陷阱和性能优化。通过图文结合的方式帮助开发者彻底理解数组本质,并提供Arrays工具类的实用方法与面试高频问题解析,助你掌握数组核心知识,避免常见错误。
|
4月前
|
Java 测试技术
Java浮点类型详解:使用与区别
Java中的浮点类型主要包括float和double,它们在内存占用、精度范围和使用场景上有显著差异。float占用4字节,提供约6-7位有效数字;double占用8字节,提供约15-16位有效数字。float适合内存敏感或精度要求不高的场景,而double精度更高,是Java默认的浮点类型,推荐在大多数情况下使用。两者都存在精度限制,不能用于需要精确计算的金融领域。比较浮点数时应使用误差范围或BigDecimal类。科学计算和工程计算通常使用double,而金融计算应使用BigDecimal。
1754 102
|
2月前
|
Java 大数据 Go
从混沌到秩序:Java共享内存模型如何通过显式约束驯服并发?
并发编程旨在混乱中建立秩序。本文对比Java共享内存模型与Golang消息传递模型,剖析显式同步与隐式因果的哲学差异,揭示happens-before等机制如何保障内存可见性与数据一致性,展现两大范式的深层分野。(238字)
81 4
|
5月前
|
存储 缓存 人工智能
Java int和Integer的区别
本文介绍了Java中int与Integer的区别及==与equals的比较机制。Integer是int的包装类,支持null值。使用==比较时,int直接比较数值,而Integer比较对象地址;在-128至127范围内的Integer值可缓存,超出该范围或使用new创建时则返回不同对象。equals方法则始终比较实际数值。
182 0
|
2月前
|
存储 缓存 Java
【深入浅出】揭秘Java内存模型(JMM):并发编程的基石
本文深入解析Java内存模型(JMM),揭示synchronized与volatile的底层原理,剖析主内存与工作内存、可见性、有序性等核心概念,助你理解并发编程三大难题及Happens-Before、内存屏障等解决方案,掌握多线程编程基石。
|
3月前
|
缓存 监控 Kubernetes
Java虚拟机内存溢出(Java Heap Space)问题处理方案
综上所述, 解决Java Heap Space溢出需从多角度综合施策; 包括但不限于配置调整、代码审查与优化以及系统设计层面改进; 同样也不能忽视运行期监控与预警设置之重要性; 及早发现潜在风险点并采取相应补救手段至关重要.
550 17
|
3月前
|
安全 Java API
Java SE 与 Java EE 区别解析及应用场景对比
在Java编程世界中,Java SE(Java Standard Edition)和Java EE(Java Enterprise Edition)是两个重要的平台版本,它们各自有着独特的定位和应用场景。理解它们之间的差异,对于开发者选择合适的技术栈进行项目开发至关重要。
430 1
|
4月前
|
监控 Kubernetes Java
最新技术栈驱动的 Java 绿色计算与性能优化实操指南涵盖内存优化与能效提升实战技巧
本文介绍了基于Java 24+技术栈的绿色计算与性能优化实操指南。主要内容包括:1)JVM调优,如分代ZGC配置和结构化并发优化;2)代码级优化,包括向量API加速数据处理和零拷贝I/O;3)容器化环境优化,如K8s资源匹配和节能模式配置;4)监控分析工具使用。通过实践表明,这些优化能显著提升性能(响应时间降低40-60%)同时降低资源消耗(内存减少30-50%,CPU降低20-40%)和能耗(服务器功耗减少15-35%)。建议采用渐进式优化策略。
214 1
|
4月前
|
存储 监控 算法
Java垃圾回收机制(GC)与内存模型
本文主要讲述JVM的内存模型和基本调优机制。