精华内容
下载资源
问答
  • 数组是静态的 Java 语言是典型的静态语言,因此 Java 数组是静态的,即当数组被初始化...所谓初始化,即创建实际的数组对象,也就是在内存中为数组对象分配内存空间,并为每个数组 元素指定初始值。 数组的初始化...

    原文:https://m.2cto.com/kf/201611/561021.html

    数组是静态的

    Java 语言是典型的静态语言,因此 Java 数组是静态的,即当数组被初始化之后,该数组 所占的内存空间、数组长度都是不可变的。Java 程序中的数组必须经过初始化才可使用。所谓初始化,即创建实际的数组对象,也就是在内存中为数组对象分配内存空间,并为每个数组 元素指定初始值。

    数组的初始化有以下两种方式。

    • 静态初始化:初始化时由程序员显式指定每个数组元素的初始值,由系统决定数组长度。
    • 动态初始化:初始化时程序员只指定数组长度,由系统为数组元素分配初始值。

    不管采用哪种方式初始化Java 数组,一旦初始化完成,该数组的长度就不可改变,Java 语言允许通过数组的length 属性来访问数组的长度。示例如下。

    public class ArrayTest
    {
    public static void main(String[] args)
    {
    // 采用静态初始化方式初始化第一个数组
    String[] books = new String[]
    { "1", "2", "3", "4"
    };
    // 采用静态初始化的简化形式初始化第二个数组
    String[] names =
    {
    "孙悟空",
    "猪八戒",
    "白骨精"
    };
    // 采用动态初始化的语法初始化第三个数组
    String[] strArr = new String[5];
    // 访问三个数组的长度
    System.out.println("第一个数组的长度:" + books.length);
    System.out.println("第二个数组的长度:" + names.length);
    System.out.println("第三个数组的长度:" + strArr.length);
    }
    }

     

    上面程序中的粗体字代码声明并初始化了三个数组。这三个数组的长度将会始终不变,程 序输出三个数组的长度依次为4 、3 、5 。

    前面已经指出,Java 语言的数组变量是引用类型的变量,books、names 、strArr 这三个变量,以及各自引用的数组在内存中的分配示意图如图1.1 所示。

    \

    从图1.1可以看出,对于静态初始化方式而言,程序员无须指定数组长度,指定该数组的 数组元素,由系统来决定该数组的长度即可。例如 books 数组,为它指定了四个数组元素,它 的长度就是4 ;对于names 数组,为它指定了三个元素,它的长度就是3 。

    执行动态初始化时,程序员只需指定数组的长度,即为每个数组元素指定所需的内存空间, 系统将负责为这些数组元素分配初始值。指定初始值时,系统将按如下规则分配初始值。

    • 数组元素的类型是基本类型中的整数类型(byte 、short、int 和long ),则数组元素的值是0 。
    • 数组元素的类型是基本类型中的浮点类型(float 、double ),则数组元素的值是0.0。
    • 数组元素的类型是基本类型中的字符类型(char ),则数组元素的值是'\u0000'。
    • 数组元素的类型是基本类型中的布尔类型(boolean),则数组元素的值是false 。
    • 数组元素的类型是引用类型(类、接口和数组),则数组元素的值是null 。

      Java 数组是静态的,一旦数组初始化完成,数组元素的内存空间分配即结束,程序只能改变数组元素的值,而无法改变数组的长度。

    需要指出的是,Java 的数组变量是一种引用类型的变量,数组变量并不是数组本身,它 只是指向堆内存中的数组对象。因此,可以改变一个数组变量所引用的数组,这样可以造成数 组长度可变的假象。假设,在上面程序的后面增加如下几行。

    // 让books 数组变量、strArr 数组变量指向names 所引用的数组
    books = names;
    strArr = names;
    System.out.println("--------------");
    System.out.println("books 数组的长度:" + books.length);
    System.out.println("strArr 数组的长度:" + strArr.length);
    // 改变books 数组变量所引用的数组的第二个元素值
    books[1] = "唐僧";
    System.out.println("names 数组的第二个元素是:" + books[1]);

     

    上面程序中粗体字代码将让books 数组变量、strArr 数组变量都指向names 数组变量所引 用的数组,这样做的结果就是books、strArr、names 这三个变量引用同一个数组对象。此时, 三个引用变量和数组对象在内存中的分配示意图如图1.2 所示。

    \

    从图1.2可以看出,此时 strArr、names 和books 数组变量实际上引用了同一个数组对象。 因此,当访问 books 数组、strArr 数组的长度时,将看到输出 3。这很容易造成一个假象:books 数组的长度从4 变成了3。实际上,数组对象本身的长度并没有发生改变,只是 books 数组变 量发生了改变。books 数组变量原本指向图 1.2下面的数组,当执行了books = names;语句之后,books 数组将改为指向图1.2 中间的数组,而原来books 变量所引用的数组的长度依然是4 。

    从图1.2 还可以看出,原来 books 变量所引用的数组的长度依然是 4 ,但不再有任何引用 变量引用该数组,因此它将会变成垃圾,等着垃圾回收机制来回收。此时,程序使用books、 names 和strArr 这三个变量时,将会访问同一个数组对象,因此把 books 数组的第二个元素赋 值为“唐僧”时,names 数组的第二个元素的值也会随之改变。

    与Java 这种静态语言不同的是,JavaScript 这种动态语言的数组长度是可以动态改变的,示例如下。

    <script type="text/javascript">
    var arr = [];
    document.writeln("arr的长度是:" + arr.length + "
    ");
    // 为arr 数组的两个数组元素赋值
    arr[2] = 6;
    arr[4] = "孙悟空";
    // 再次访问arr 数组的长度
    document.writeln("arr的长度是:" + arr.length + "
    ");
    </script>

     

    上面是一个简单的JavaScript 程序。它先定义了一个名为 arr的空数组,因为它不包含任 何数组元素,所以它的长度是0 。接着,为 arr数组的第三个、第五个元素赋值,该数组的长 度也自动变为5 。这就是JavaScript 里动态数组和Java 里静态数组的区别。

    基本类型数组的初始化

    对于基本类型数组而言,数组元素的值直接存储在对应的数组元素中,因此基本类型 数组的初始化比较简单:程序直接先为数组分配内存空间,再将数组元素的值存入对应内 存里。

    下面程序采用静态初始化方式初始化了一个基本类型的数组对象。

    public class PrimitiveArrayTest
    {
    public static void main(String[] args)
    {
    // 定义一个int[] 类型的数组变量
    int[] iArr;
    // 静态初始化数组,数组长度为4
    iArr = new int[]{2 , 5 , -12 , 20};
    }
    }

     

    上面代码的执行过程代表了基本类型数组初始化的典型过程。下面将结合示意图详细介绍这段代码的执行过程。

    执行第一行代码int[] iArr;时,仅定义一个数组变量,此时内存中的存储示意图如图1.4所示。

    \

    执行了int[] iArr; 代码后,仅在 main 方法栈中定义了一个 iArr 数组变量,它是一个引用类 型的变量,并未指向任何有效的内存,没有真正指向实际的数组对象。此时还不能使用该数组 对象。

    当执行iArr = new int[]{2,5,-12,20}; 静态初始化后,系统会根据程序员指定的数组元素来决 定数组的长度。此时指定了四个数组元素,系统将创建一个长度为4 的数组对象,一旦该数组 对象创建成功,该数组的长度将不可改变,程序只能改变数组元素的值。此时内存中的存储示 意图如图1.5 所示。

    静态初始化完成后,iArr 数组变量引用的数组所占用的内存空间被固定下来,程序员只能 改变各数组元素内的值。既不能移动该数组所占用的内存空间,也不能扩大该数组对象所占用 的内存,或缩减该数组对象所占用的内存。

    \

    对于程序运行过程中的变量,可以将它们形容为具体的瓶子——瓶子可以存储 水,而变量用于存储值,也就是数据。对于强类型语言如Java ,它有一个要求: 怎样的瓶子只能装怎样的水,也就是说,指定类型的变量只能存储指定类型的值。

    所有局部变量都是放在栈内存里保存的,不管其是基本类型的变量,还 是引用类型的变量,都是存储在各自的方法栈内存中的;但引用类型的变量所引用的对象(包 括数组、普通的Java 对象)则总是存储在堆内存中。

    对于Java 语言而言,堆内存中的对象(不管是数组对象,还是普通的 Java 对象)通常不允许直接访问,为了访问堆内存中的对象,通常只能通过引用变量。这也是很容易混淆的地方。 例如,iArr 本质上只是main 栈区的引用变量,但使用 iArr.length 、iArr[2] 时,系统将会自动变 为访问堆内存中的数组对象。

    对于很多Java 程序员而言,他们最容易混淆的是:引用类型的变量何时只是栈内存中的 变量本身,何时又变为引用实际的Java 对象。其实规则很简单:引用变量本质上只是一个指 针,只要程序通过引用变量访问属性,或者通过引用变量来调用方法,该引用变量就会由它所 引用的对象代替。

    public class PrimitiveArrayTest2
    {
    public static void main(String[] args)
    {
    // 定义一个int[] 类型的数组变量
    int[] iArr = null;
    // 只要不访问iArr 的属性和方法,程序完全可以使用该数组变量
    System.out.println(iArr); //①
    // 动态初始化数组,数组长度为5
    iArr = new int[5];
    // 只有当iArr 指向有效的数组对象后,下面才可访问iArr 的属性
    System.out.println(iArr.length); //②
    }
    }

     

    上面程序中两行粗体字代码两次访问iArr 变量。对于①行代码而言,虽然此时的iArr 数 组变量并未引用到有效的数组对象,但程序在①行代码处并不会出现任何问题,因为此时并未 通过iArr 访问属性或调用方法,因此程序只是访问iArr 引用变量本身,并不会去访问iArr 所 引用的数组对象。对于②行代码而言,此时程序通过iArr 访问了length 属性,程序将自动变 为访问iArr 所引用的数组对象,这就要求iArr 必须引用一个有效的对象。

    有过一些编程经验,应该经常看到一个Runtime 异常: NullPointerException (空指针异常)。当通过引用变量来访问实例属性,或者调 用非静态方法时,如果该引用变量还未引用一个有效的对象,程序就会引发 NullPointerException 运行时异常。

    引用类型数组的初始化

    引用类型数组的数组元素依然是引用类型的,因此数组元素里存储的还是引用,它指向另一块内存,这块内存里存储了该引用变量所引用的对象(包括数组和Java 对象)。

    为了说明引用类型数组的运行过程,下面程序先定义一个Person 类,然后定义一个 Person[]数组,并动态初始化该Person[]数组,再显式地为数组的不同数组元素指定值。该程序代码如下。

    class Person
    {
    // 年龄
    public int age;
    // 身高
    public double height;
    // 定义一个info 方法
    public void info()
    {
    System.out.println("我的年龄是:" + age
    + ",我的身高是:" + height);
    }
    }
    public class ReferenceArrayTest
    {
    public static void main(String[] args)
    {
    // 定义一个students 数组变量,其类型是Person[]
    Person[] students;
    // 执行动态初始化
    students = new Person[2];
    System.out.println("students所引用的数组的长度是:"
    + students.length); //①
    // 创建一个Person 实例,并将这个Person 实例赋给zhang 变量
    Person zhang = new Person();
    // 为zhang 所引用的Person 对象的属性赋值
    zhang.age = 15;
    zhang.height = 158;
    // 创建一个Person 实例,并将这个Person 实例赋给lee 变量
    Person lee = new Person();
    // 为lee 所引用的Person 对象的属性赋值
    lee.age = 16;
    lee.height = 161;
    // 将zhang 变量的值赋给第一个数组元素
    students[0] = zhang;
    // 将lee 变量的值赋给第二个数组元素
    students[1] = lee;
    // 下面两行代码的结果完全一样,
    // 因为lee 和students[1]指向的是同一个Person 实例
    lee.info();
    students[1].info();
    }
    }

     

    上面代码的执行过程代表了引用类型数组的初始化的典型过程。下面将结合示意图详细介绍这段代码的执行过程。

    执行Person[] students;代码时,这行代码仅仅在栈内存中定义了一个引用变量,也就是一个指针,这个指针并未指向任何有效的内存区。此时内存中的存储示意图如图1.6 所示。

    \

    在图1.6中的栈内存中定义了一个 students 变量,它仅仅是一个空引用,并未指向任何有 效的内存,直到执行初始化,本程序对 students 数组执行动态初始化。动态初始化由系统为数 组元素分配默认的初始值null ,即每个数组元素的值都是 null 。执行动态初始化后的存储示意 图如图1.7 所示。

    从图1.7 中可以看出,students 数组的两个数组元素都是引用,而且这两个引用并未指 向任何有效的内存,因此,每个数组元素的值都是 null 。此时,程序可以通过students 来 访问它所引用的数组的属性,因此在①行代码处通过 students 访问了该数组的长度,此时 将输出2 。

    students 数组是引用类型的数组,因此 students[0] 、students[1] 两个数组元素相当于两个引 用类型的变量。如果程序只是直接输出这两个引用类型的变量,那么程序完全正常。但程序依 然不能通过students[0] 、students[1] 来调用属性或方法,因此它们还未指向任何有效的内存区, 所以这两个连续的Person 变量(students 数组的数组元素)还不能被使用。

    \

    接着,程序定义了zhang 和lee 两个引用变量,并让它们指向堆内存中的两个Person 对象,此时的zhang、lee 两个引用变量存储在 main 方法栈区中,而两个 Person 对象则存储在堆内存中。此时的内存存储示意图如图1.8 所示。

    \

    对于zhang、lee 两个引用变量来说,它们可以指向任何有效的Person 对象,而students[0] 、 students[1] 也可以指向任何有效的Person 对象。从本质上来看,zhang、lee、students[0] 、students[1] 能够存储的内容完全相同。接着,程序执行students[0] = zhang;和students[1] = lee; 两行代码, 也就是让zhang 和students[0] 指向同一个 Person 对象,让 lee 和students[1] 指向同一个Person 对象。此时的内存存储示意图如图1.9 所示。

    \

    从图1.9 中可以看出,此时 zhang 和students[0] 指向同一个内存区,而且它们都是引用类 型的变量,因此通过 zhang 和students[0] 来访问Person 实例的属性和方法的效果完全一样。不 论修改students[0] 所指向的 Person 实例的属性,还是修改 zhang 变量所指向的 Person 实例的 属性,所修改的其实是同一个内存区,所以必然互相影响。同理,lee 和students[1] 也是引用 到同一个Person 对象,也有相同的效果。

    前面已经提到,对于引用类型的数组而言,它的数组元素其实就是一个引用类型的变量, 因此可以指向任何有效的内存——此处“有效”的意思是指强类型的约束。比如,对 Person[] 类型的数组而言,它的每个数组元素都相当于Person 类型的变量,因此它的数组元素只能指 向Person 对象。

     
     
     
    展开全文
  • java数组内存分配

    千次阅读 2020-07-04 17:40:39
    声明一个长度为3的int类型数组:int[] arr=new int[3] 栈内存中存放局部变量,定义在方法中的变量,...数组在初始化时会为存储空间分配默认值。 整数:0 浮点数:0.0 字符:空字符 布尔:false 引用数据类型:null ...

    声明一个长度为3的int类型数组:int[] arr=new int[3]
    在这里插入图片描述栈内存中存放局部变量,定义在方法中的变量,如arr。使用完毕立即消失。
    堆内存存放new出来的内容(实体、对象),每一个new出来的内容都有地址值。使用完毕会在垃圾回收机制空闲时被回收。
    数组在初始化时会为存储空间分配默认值。
    整数:0
    浮点数:0.0
    字符:空字符
    布尔:false
    引用数据类型:null

    展开全文
  • Java数组内存分配 关于数组变量: 本质上,数组变量是数组的管理者而非数组本身; 数组变量之间的赋值是管理权限的赋予; 数组变量之间的比较是判断是否管理同一个数组。 所以要由一个数组得到一个和它完全...

    Java中数组的内存分配

    关于数组变量:

    • 本质上,数组变量是数组的管理者而非数组本身;
    • 数组变量之间的赋值是管理权限的赋予;
    • 数组变量之间的比较是判断是否管理同一个数组。

    所以要由一个数组得到一个和它完全一样的数组,要采用遍历赋值的方法。

    Java一维数组的内存分配

    堆区中分配空间后会把每个数组元素初始化为0。array是引用变量,它在栈区占用的空间大小为4bytes
    在这里插入图片描述
    由此可以看出,数组变量其实就是指针。它存储着数组元素在堆区的首地址。

    Java中对象数组的内存分配

    创建对象数组:Department[] dept = new Department[100];
    dept是引用变量。
    内存分配关键在于new
    对象数组的默认初始化全部是null,也就是不指向任何对象。所以想要真正初始化,就要:dept[0] = new Department();
    这一内存分配模型也适用于二维数组。
    在这里插入图片描述
    从图中可以看出,对象数组变量其实就是指针的指针

    展开全文
  • Java中,声明数组的语法有两种: 1.数据类型 数组名[]; 2.数据类型[] 数组名; 例如: int a[]; 与 int[] a;都是表示声明了一个整型数组a 二维数组的声明也类似。 来看看二维数组分配方式: 例如:int...

    Java中,声明数组的语法有两种:

    1.数据类型 数组名[];

    2.数据类型[] 数组名;

    例如:

    int a[]; 与 int[] a;都是表示声明了一个整型数组a

    二维数组的声明也类似。


    来看看二维数组存分配方式:

    例如:int a[][] = new int[2][3];

    其内存空间分配示意图如下:


    我画的图很丑,我想表示的意思是要注意:a.length是等于2,而不是2*3


    再来个示例代码说明一下:

    		int a[] = new int[]{10,20,30,40,50};
    		System.out.println("a length:"+a.length);
    		for(int i:a){
    			System.out.println(i);
    		}
    		
    		int b[][] = new int[3][4];
    		System.out.println("b length:"+b.length);
    		for(int i=0;i<b.length;i++){
    			for(int j=0;j<b[i].length;j++){
    				System.out.println("b["+i+"] length:"+b[i].length);
    			}
    		}
    
    		int c[][] = new int[3][];
    		c[0] = new int[2];
    		c[1] = new int[3];
    		c[2] = new int[2];
    		System.out.println("c length:"+c.length);
    		for(int i=0;i<c.length;i++){
    			for(int j=0;j<c[i].length;j++){
    				System.out.println("c["+i+"] length:"+c[i].length);
    			}
    		}
    输出结果:
    a length:5
    10
    20
    30
    40
    50
    b length:3
    b[0] length:4
    b[0] length:4
    b[0] length:4
    b[0] length:4
    b[1] length:4
    b[1] length:4
    b[1] length:4
    b[1] length:4
    b[2] length:4
    b[2] length:4
    b[2] length:4
    b[2] length:4
    c length:3
    c[0] length:2
    c[0] length:2
    c[1] length:3
    c[1] length:3
    c[1] length:3
    c[2] length:2
    c[2] length:2
    




    展开全文
  • Java数组分配内存空间

    2018-06-25 15:57:00
    分配内存空间 ...定义数组+分配内存空间 数据类型[]数组名=new 数据类型[数组长度]; 定义数组时不指定长度,分配空间时指定数组长度;如:String cities[] = new String[6]; 数组元素: 数...
  • Java数组及其内存分配

    千次阅读 2016-08-30 10:22:25
    几乎所有的程序设计语言都支持数组。Java也不例外。当我们需要多个类型相同的变量的时候,就考虑定义一个数组。...对于Java数组的初始化,有以下两种方式,这也是面试中经常考到的经典题目: 静态初始
  • Java数组内存分配

    2020-09-11 23:31:36
    Java数组内存分配 一.Java数组 对于Java数组的初始化,主要有三种方式 1.静态初始化 在数组定义的时候直接为数组赋值,将内容放入为其申请的空间中 int[] a={1,2,3,4}//数组a一旦定义长度就不可改变; String[] ...
  • 关于Java 数组内存分配一点认识

    千次阅读 2013-05-16 23:07:14
    可能Java 数组大家都很熟悉,最近我遇到了一个关于Java 数组内存分配的问题。  呵呵。突然就发现许多书上“基本数据类型存储在栈内存当中,对象则保存在堆内存”这句话完全是错误的。下面是个简单的例子代码: ...
  • 想查看以下java是如何分配内存的,但是没有发现java有相应的函数供使用 google之后关于这两个问题的解答: 1、java数组的内存并不是连续分配的,把二维数组看成数组的数组,也就是一个一维数组,每个数组的元素...
  • Java数组内存中的分配

    千次阅读 2019-02-21 23:29:52
    Java数组内存中的分配   在Java中,数组存储对象的原始值(int,char,...)或引用(也称为指针)。 使用“new”创建对象时,会在堆中分配一个内存空间并返回一个引用。对于数组也是如此,因为数组是Java中的...
  • java --数组内存分配等问题
  • 如何实现动态分配? 说白了其实就是数组大小由外部输入,然后在动态生成数组。 因此在添加元素时需要判断数组是否已经满员了,所以类中需要一个记录了数组大小的标记,这里记为font 我以下的代码写了一个Array的父类...
  • 刚看了一篇关于逃逸分析的博客说:Java数组长度小于等于64会是在栈分配的,大于64的会分配在堆里面。 问题1:这个是真的吗?官方文档哪里有些? 问题2:如果是真的,那么基础数据类型数组和对象数组都一样在小于64在...
  • 1.Java数组是静态的 Java是静态语言,所以Java的数组也是静态的,即:数组被初始化后,长度不可变 静态初始化:显式指定每个数组元素的初始值,系统决定数组长度 String[] books = new String[]{"疯狂Java讲义",...
  • Java数组内存分配 1、Java 程序在运行时,需要在内存分配空间。为了提高运算效率,就对空间进行了不同区域的划分,因为每一片区域都有特定的处理数据和内存管理方式。 2、数组基本概念 数组是存储同一种数据...
  • 数组
  • 同C或C++不同,java在定义数组时不分配空间,所以不必指出数组大小,在new时才会分配空间,必须指明数组大小。 数组有一个length属性,通过数组名.length调用。arrayName=new type[arraySize];在定义对象时同理例如...
  • 关于数组,我们首先要知道的是,他是引用数据类型,数组是存储同一种数据类型多个元素的容器。...内存,简单说就是存放正在运行的程序,我们知道,java程序运行的载体是JVM,运行环境是JRE,内存分配给JVM空间执...
  • type为java中的任意数据类型,包括基本类型和组合类型,arrayName为数组名,必须是一个合法的标识符,[]指明该变量是一个数组类型变量 例如: int demoArray []; int[] demoArray;这两种形式没有区别,
  • Java数组内存管理

    2013-07-31 17:49:35
    Java数组内存管理
  • Java多维数组内存分配

    千次阅读 2017-01-16 03:48:19
    什么是多维数组多维数组的本质还是1维数组,就是数组元素依然是数组...示例代码(下面的代码只是为了讲解多维数组内存分配机制,实际中不会写出这么绕的多维数组) public class ObjectArrayTest{ public static v
  • [精]JAVA数组内存结构详解

    万次阅读 2016-10-31 14:07:32
    所 谓初始化,即创建实际的数组对象,也就是在内存中为数组对象分配内存空间,并为每个数组 元素指定初始值。数组的初始化有以下两种方式。静态初始化:初始化时由程序员显式指定每个数组元素的初始值,由系统决定数
  • 数组名即引用在Java栈空间,数组元素在堆,并且空间连续。正式因为内存地址空间连续,访问数组元素时只需要根据起始地址和元素偏移量即可快速访问数组中的任意元素。 如果数组元素是基本类型,数组存放的是基本类型...
  • 一、数组初始化Java 中的数组变量是引用类型的变量。Java 是静态语言,因此 Java 数组是静态的,即当数组初始化...动态初始化:初始化时只指定数组长度,由系统分配初始值。 public static void main(String[] args) {

空空如也

空空如也

1 2 3 4 5 ... 20
收藏数 185,106
精华内容 74,042
关键字:

java数组动态分配内存

java 订阅