Java | 如何从内存解析的角度理解“数组名实质是一个地址”?

本文涉及的产品
公共DNS(含HTTPDNS解析),每月1000万次HTTP解析
云解析 DNS,旗舰版 1个月
全局流量管理 GTM,标准版 1个月
简介: 这篇文章讨论了Java内存的简化结构以及如何解析一维和二维数组的内存分配。在Java中,内存分为栈和堆,栈存储局部变量,堆存储通过`new`关键字创建的对象和数组。方法区包含静态域和常量池。文章通过示例代码解释了一维数组的创建过程,分为声明数组、分配空间和赋值三个步骤,并提供了内存解析图。接着,介绍了二维数组的内存解析,强调二维数组是“数组的数组”,其内存结构中,外层元素存储内层数组的地址。最后,文章提到了默认初始化方式对初始值的影响,并给出了相关测试代码。

一、内存的简化结构

 

下图即为内存的简化结构。在Java语言中,内存的存储分配是这样的:


栈:局部变量

堆:new出来的东西,如对象、数组等


方法区:包括静态域(static)和常量池(String的内容就存储在这里)



内存的简化结构


这张内存简化图非常重要,需要大家留有印象。


接下来我们在该图和结论的基础上,分步来看一维数组与二维数组的内存解析。


二、一维数组的内存解析


1. 分步解析


示例代码


public class Test{
    public static void main(String args[]){
        int[] arr;   
        arr = new int[10];
 
        for ( int i=0; i<10; i++ ) {  
            arr[i] =2*i+1;
            System.out.println(arr[i]);
        }
    }
}


step1  int[] arr;


此时在栈中创建出了变量arr



step2  arr = new int[10];


接着使用new关键字,来创建一个一维数组。需要注意的是,基本数据类型数组在显式赋值之前, Java会自动给它们赋默认值。由于一维数组的每个元素都是int类型,因而默认值为0.



step3  arr[i] =2*i+1;  


在for循环中遍历数组arr并为其元素赋值:



综合起来看,内存状态如下:

 

2. 综合解析


如图,创建数组并赋值的过程可以简化成如下示意图。



左侧为栈区,右侧为堆区


当声明数组int[ ] arr 时,arr属于局部变量,在栈中创建。如上图中,int[ ] arr1和String[ ] arr2的操作执行后,在左侧的栈中创建了变量arr1与arr2. 若没有new的操作,实际上还未给数组开辟存储空间。


只有当通过new关键字创建数组对象后,系统才在堆中划分相应的存储空间,并依据数组元素的数据类型给新划分的空间自动赋初值。如上图中,在new int[4]后,堆区开辟了4个连续的存储空间,并赋值为0,而new String[3]后,堆区又开辟了3个存储空间用来存储String类型的数据,引用类型String默认初值为null.


同时,数组的地址(即数组首元素的地址,假设为0x12ab)赋值给栈区中的变量arr1,即变量arr1中存着数组的地址,通过该地址,arr1可以轻松地在堆中找到它对应的数组。


我们通过中括号 [ ] 来访问数组中的各个元素。我们执行arr1[0] = 10;这一代码时,实际便是根据栈内arr1中存着的地址,找到将堆区中的数组空间,并将第一个空间中的0改为了10.


而对于arr2[ ]数组,在new过之后又new了一次,第二次通过new String[5]开辟了一片5个存储空间的数组。这时变量arr2中原本存着的地址0x34ab被新地址0x78cd覆盖,arr2变量存放了新数组的地址。原数组在后续的某个时间内,将被自动回收。


当然,当栈区存有数组地址的变量arr1与变量arr2最终出栈后,在堆区划分的数组空间也将在之后被回收。


三、二(多)维数组的内存解析


1. 综合解析


二维数组是“数组的数组”,即一个一维数组中每个元素也是数组。由于数组既可以存储基本数据类型,也可以存储引用数据类型,因而“数组中存数组”的理解是可行的。


二维数组的创建过程与一维数组类似,这里便不再分步解析。我们直接来看内存解析图:



二维数组的内存解析


与一维数组的不同之处在于,二维数组中外层元素也用于存储地址。在创建二维数组时,除了在堆区中为外层元素(一维数组)开辟了存储空间外,还为内层元素开辟了存储空间。同时,内层元素的首元素地址返回给外层元素。


这样,数组名arr1与一维数组名arr1[ ] 像桥梁一样连接到内层元素arr1[ ][ ]。通过地址和存有地址的栈区变量arr1、堆区中的一维数组arr1[ ],我们可以轻松地访问到内层元素。


其中,内层元素的长度可以不相等。int[ ][ ] arr = new int[ ][ ]{{1,2,3},{4,5},{6,7,8}};这样也是可行的,从图中就能清晰的看出,内层元素之间其实是相对独立的。


2. 默认初始化方式对初始值的影响


针对于 int[][] arr = new int[4][3]; 这样将内外层元素个数都指定了的初始化方式,外层元素的初始化值为地址值,内层元素的初始化值则与一维数组初始化情况相同(由元素的数据类型而决定)。


而针对 int[][] arr = new int[4][]; 这样省略内层元素长度的初始化方式而言,外层元素的初始化值为null(相当于未赋值的引用数据类型),内层元素则根本没有初始化值而言(空间还没开辟),不能调用,否则编译器将报错。


测试


//测试代码
 
public class ArrayTest {
  public static void main(String[] args) {
    
    int[][] arr = new int[4][3];
    System.out.println(arr[0]); //[I@15db9742
    System.out.println(arr[0][0]);  //0
    
        //System.out.println(arr);
    
    System.out.println("***********************");
    float[][] arr1 = new float[4][3];
    System.out.println(arr1[0]);  //地址值
    System.out.println(arr1[0][0]); //0.0
    
    System.out.println("***********************");
    
    String[][] arr2 = new String[4][2];
    System.out.println(arr2[1]);  //地址值
    System.out.println(arr2[1][1]); //null
    
    System.out.println("*********************");
    double[][] arr3 = new double[4][];
    System.out.println(arr3[1]);  //null
//    System.out.println(arr3[1][0]); //报错
  }
}


总结


单看数组名,实际上是一个创建在栈区的局部变量。 整个数组数据量可能较大,直接把数组内所有的元素都存放在栈区是不太妥当的。因而,数组的主体部分实际上开辟在堆区。


要想访问数组,若堆区的内容与栈区的数组名arr之间没有任何关系,是无法找到想要访问的数组内容的。因而,堆区会返回开辟的数组空间首元素的地址作为数组主体的地址,传给栈区的局部变量arr(它是一个数组名)。若是二维数组,内层元素的首元素地址会作为外层元素(也就是一维数组名)的内容。


通过地址和存储地址的“数组名”、“一维数组名”,我们可以一连串地找到我们想要访问的数组元素。


该部分内容我用文字表述可能不够简练,大家更多地可以看图,图为重点,通过图示来理解内存解析更好一些。


相关文章
|
17天前
|
安全 Java 程序员
深入理解Java内存模型与并发编程####
本文旨在探讨Java内存模型(JMM)的复杂性及其对并发编程的影响,不同于传统的摘要形式,本文将以一个实际案例为引子,逐步揭示JMM的核心概念,包括原子性、可见性、有序性,以及这些特性在多线程环境下的具体表现。通过对比分析不同并发工具类的应用,如synchronized、volatile关键字、Lock接口及其实现等,本文将展示如何在实践中有效利用JMM来设计高效且安全的并发程序。最后,还将简要介绍Java 8及更高版本中引入的新特性,如StampedLock,以及它们如何进一步优化多线程编程模型。 ####
21 0
|
20天前
|
存储 监控 算法
Java内存管理深度剖析:从垃圾收集到内存泄漏的全面指南####
本文深入探讨了Java虚拟机(JVM)中的内存管理机制,特别是垃圾收集(GC)的工作原理及其调优策略。不同于传统的摘要概述,本文将通过实际案例分析,揭示内存泄漏的根源与预防措施,为开发者提供实战中的优化建议,旨在帮助读者构建高效、稳定的Java应用。 ####
32 8
|
17天前
|
存储 监控 算法
深入探索Java虚拟机(JVM)的内存管理机制
本文旨在为读者提供对Java虚拟机(JVM)内存管理机制的深入理解。通过详细解析JVM的内存结构、垃圾回收算法以及性能优化策略,本文不仅揭示了Java程序高效运行背后的原理,还为开发者提供了优化应用程序性能的实用技巧。不同于常规摘要仅概述文章大意,本文摘要将简要介绍JVM内存管理的关键点,为读者提供一个清晰的学习路线图。
|
22天前
|
存储 算法 Java
Java 内存管理与优化:掌控堆与栈,雕琢高效代码
Java内存管理与优化是提升程序性能的关键。掌握堆与栈的运作机制,学习如何有效管理内存资源,雕琢出更加高效的代码,是每个Java开发者必备的技能。
48 5
|
20天前
|
存储 算法 Java
Java内存管理深度解析####
本文深入探讨了Java虚拟机(JVM)中的内存分配与垃圾回收机制,揭示了其高效管理内存的奥秘。文章首先概述了JVM内存模型,随后详细阐述了堆、栈、方法区等关键区域的作用及管理策略。在垃圾回收部分,重点介绍了标记-清除、复制算法、标记-整理等多种回收算法的工作原理及其适用场景,并通过实际案例分析了不同GC策略对应用性能的影响。对于开发者而言,理解这些原理有助于编写出更加高效、稳定的Java应用程序。 ####
|
20天前
|
安全 Java 程序员
Java内存模型的深入理解与实践
本文旨在深入探讨Java内存模型(JMM)的核心概念,包括原子性、可见性和有序性,并通过实例代码分析这些特性在实际编程中的应用。我们将从理论到实践,逐步揭示JMM在多线程编程中的重要性和复杂性,帮助读者构建更加健壮的并发程序。
|
20天前
|
存储 监控 算法
Java虚拟机(JVM)垃圾回收机制深度解析与优化策略####
本文旨在深入探讨Java虚拟机(JVM)的垃圾回收机制,揭示其工作原理、常见算法及参数调优方法。通过剖析垃圾回收的生命周期、内存区域划分以及GC日志分析,为开发者提供一套实用的JVM垃圾回收优化指南,助力提升Java应用的性能与稳定性。 ####
|
16天前
|
存储 监控 算法
Java内存管理的艺术:深入理解垃圾回收机制####
本文将引领读者探索Java虚拟机(JVM)中垃圾回收的奥秘,解析其背后的算法原理,通过实例揭示调优策略,旨在提升Java开发者对内存管理能力的认知,优化应用程序性能。 ####
31 0
|
1月前
|
监控 Java 应用服务中间件
高级java面试---spring.factories文件的解析源码API机制
【11月更文挑战第20天】Spring Boot是一个用于快速构建基于Spring框架的应用程序的开源框架。它通过自动配置、起步依赖和内嵌服务器等特性,极大地简化了Spring应用的开发和部署过程。本文将深入探讨Spring Boot的背景历史、业务场景、功能点以及底层原理,并通过Java代码手写模拟Spring Boot的启动过程,特别是spring.factories文件的解析源码API机制。
71 2
|
2月前
|
缓存 Java 程序员
Map - LinkedHashSet&Map源码解析
Map - LinkedHashSet&Map源码解析
76 0

推荐镜像

更多