JAVA数组

数组的定义与使用

数组创建及初始化

数组创建

T[] 数组名 = new T[N];

T:数组元素类型

T[ ]:表示为数组类型

N:表示数组的长度

 int[] array1=new int[10];    //创建一个可容纳10个int类型元素的数组
 double[] array2 = new double[5];//创建一个可容纳5个double类型元素的数组
 String[] array3 = new String[3];//创建一个可以容纳3个字符串元素的数组

数组初始化

动态初始化

创建数组时,直接指定数组中元素的个数

int[] array = new int[10];
int[] array1=new int[array.length];

静态初始化

创建数组时,不直接指定数据元素个数,直接指定具体的数据内容

int[] array1= new int[]{1,2,3,4,5,6};
int[] array2={1,2,3,4};    //省去new T[],编译器编译时会还原

注意事项

  • 静态初始化虽然没有指定数组的长度,编译器在编译时会根据{}中元素个数来确定数组的长度。
  • 静态初始化时, {}中数据类型必须与[]前数据类型一致。
  • 静态初始化可以简写,省去后面的new T[]。
  • 数组也可以按照如下C语言个数创建,不推荐
/*
该种定义方式不太友好,容易造成数组的类型就是int的误解
[]如果在类型之后,就表示数组类型,因此int[]结合在一块写意思更清晰
*/
int arr[] = {1, 2, 3};
  • 静态和动态初始化也可以分为两步,但是省略格式不可以
int[] array1;
array1 = new int[10];

int[] array2;
array2 = new int[]{10, 20, 30};
// 注意省略格式不可以拆分, 否则编译失败
// int[] array3;
// array3 = {1,2,3};
  • 如果没有对数组进行初始化,数组中元素有其默认值
    • 如果数组中存储元素类型为基类类型,默认值为基类类型对应的默认值,比如:

类型

默认值

byte

0

short

0

int

0

long

0

float

0.0f

double

0.0

char

\u0000

boolean

false

  •  数组存储类型为引用类型,默认为null

数组使用

数组元素访问

下标访问,同C语言

int[] arr=new int[]{1,2,3};
arr[0];
arr[1];
arr[2];

注意事项

        数组为一段连续的内存空间,支持随机访问(通过下标)

        下标从0开始 [ 0 , N ),N为元素个数,不能越界,越界报异常

遍历数组

循环遍历--通过下标

同C语言

int[] arr=new int[]{1,2,3};
for(int i=0;i<arr.length;++i){
    System.out.println(arr[i]);
}

强循环遍历

for-each

for(Type data : 容器){
    代码;
}
int[] arr=new int[]{1,2,3};
for(int x:arr){
    System.out.println(arr[i]);
}

for-each 是 for 循环的另外一种使用方式. 能够更方便的完成对数组的遍历. 可以避免循环条件和更新语句写错

注意事项

  • 循环遍历,防止越界
  • 涉及下标,使用循环遍历

数组是引用类型

JVM的内存分布

程序计数器 (PC Register): 只是一个很小的空间, 保存下一条执行的指令的地址

虚拟机栈(JVM Stack): 与方法调用相关的一些信息,每个方法在执行时,都会先创建一个栈帧,栈帧中包含有: 局部变量表、操作数栈、动态链接、返回地址以及其他的一些信息,保存的都是与方法执行时相关的一些信 息。比如:局部变量。当方法运行结束后,栈帧就被销毁了,即栈帧中保存的数据也被销毁了。

本地方法栈(Native Method Stack): 本地方法栈与虚拟机栈的作用类似. 只不过保存的内容是Native方法的局部 变量. 在有些版本的 JVM 实现中(例如HotSpot), 本地方法栈和虚拟机栈是一起的

堆(Heap): JVM所管理的最大内存区域. 使用 new 创建的对象都是在堆上保存 (例如前面的 new int[]{1, 2, 3} ),堆是随着程序开始运行时而创建,随着程序的退出而销毁,堆中的数据只要还有在使用,就不会被销毁。

方法区(Method Area): 用于存储已被虚拟机加载的类信息、常量、静态变量、即时编译器编译后的代码等数 据. 方法编译出的的字节码就是保存在这个区域

基本数据类型与引用类型变量的区别

基本数据类型创建的变量,称为基本变量,该变量空间中直接存放的是其所对应的值;

引用数据类型创建的变量,一般称为对象的引用,其空间中存储的是对象所在空间的地址

Java中实际地址不可得,更具实际地址通过一定算法,得出伪地址(唯一值)

public static void func(){
    int a =10;
    int b=20;
    int[] arr = new int[]{1,2,3};
}

在上述代码中,a、b、arr,都是函数内部的变量,因此其空间都在main方法对应的栈帧中分配。

a、b是内置类型的变量,因此其空间中保存的就是给该变量初始化的值。

array是数组类型的引用变量,其内部保存的内容可以简单理解成是数组在堆空间中的首地址。

引用变量并不直接存储对象本身,可以简单理解成存储的是对象在堆中空间的起始地址。通过该地址,引用变量便可以去操作对象。有点类似C语言中的指针,但是Java中引用要比指针的操作更简单

再谈引用变量

public static void func() {
    int[] array1 = new int[3];
    array1[0] = 10;
    array1[1] = 20;
    array1[2] = 30;
    
    int[] array2 = new int[]{1,2,3,4,5};
    array2[0] = 100;
    array2[1] = 200;
    
    array1 = array2;
    array1[2] = 300;
    array1[3] = 400;
    array2[4] = 500;
    for (int i = 0; i < array2.length; i++) {
        System.out.println(array2[i]);
    }
}

认识null

null在Java中表示“空引用”,就是不指向任何对象的引用

int[] arr = null;
System.out.println(arr[0]);

// 执行结果
Exception in thread "main" java.lang.NullPointerException
    at Test.main(Test.java:6)

null 的作用类似于 C 语言中的 NULL (空指针), 都是表示一个无效的内存位置. 因此不能对这个内存进行任何读写操作.一旦尝试读写, 就会抛出 NullPointerException.

注意:

Java中并没有约定null和0号地址的内存有任何关联

在C语言中null为0号地址而号地址收到保护不能对这个内存进行任何读写操作.

数组应用场景

保存数据

public static void main(String[] args) {
    int[] array = {1, 2, 3};
    for(int i = 0; i < array.length; ++i){
        System.out.println(array[i] + " ");
    }
}

作为函数的参数

传基本参数

public static void main(String[] args) {
    int num = 0;
    func(num);
    System.out.println("num = " + num);
}
p
ublic static void func(int x) {
    x = 10;
    System.out.println("x = " + x);
}
/
/ 执行结果
x = 10
num = 0

发现在func方法中修改形参 x 的值, 不影响实参的 num 值

参数传数组类型(引用数据类型)

public static void main(String[] args) {
    int[] arr = {1, 2, 3};
    func(arr);
    System.out.println("arr[0] = " + arr[0]);
}

public static void func(int[] a) {
    a[0] = 10;
    System.out.println("a[0] = " + a[0]);
} 

// 执行结果
a[0] = 10
arr[0] = 10

总结: 所谓的 "引用" 本质上只是存了一个地址. Java 将数组设定成引用类型, 这样的话后续进行数组参数传参, 其实只是将数组的地址传入到函数形参中. 这样可以避免对整个数组的拷贝(数组可能比较长, 那么拷贝开销就会很大).

作为函数的返回值

public class TestArray {
    public static int[] fib(int n){
        if(n <= 0){
            return null;
        } 
        i
nt[] array = new int[n];
        array[0] = array[1] = 1;
        for(int i = 2; i < n; ++i){
            array[i] = array[i-1] + array[i-2];
        } 
        r
eturn array;
    } 
    p
ublic static void main(String[] args) {
        int[] array = fib(10);
        for (int i = 0; i < array.length; i++) {
            System.out.println(array[i]);
        }
    }
}

二维数组

二维数组本质上也就是一维数组, 只不过每个元素又是一个一维数组.

基本语法

数据类型[][] 数组名称 = new 数据类型 [][] { 初始化数据 };
数据类型[][] 数组名称 = new 数据类型 [][]; //第一个不可省略

示例

int[][] arr = {
    {1, 2, 3, 4},
    {5, 6, 7, 8},
    {9, 10, 11, 12}
};
//错误
//int[][] arr = {1,2,3,4,5,6
};

for (int row = 0; row < arr.length; row++) {
    for (int col = 0; col < arr[row].length; col++) {
        System.out.printf("%d\t", arr[row][col]);
    } 
    S
ystem.out.println("");
} 
/
/ 执行结果
1 2 3 4
5 6 7 8
9 10 11 12

最难不过坚持!

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值