Java中的数组——详解!

Java中的数组——详解!

一、数组的概念与基本要素

1.概念

数组(array)是一种最简单的复合数据类型,它是有序数据的集合,数组中的每个元素具有相同的数据类型,可以用一个统一的数组名和不同的下标来确定数组中唯一的元素。根据数组的维度,可以将其分为一维数组、二维数组和多维数组等。

2.基本要素

一个数组由4个基本元素构成:数组名称、数组元素、元素索引、数据类型。

数组的索引就相当于C语言中数组的下标,关于数组的长度可以用数组名.length求出。

int[] array = {1,2,3};
int sz = arr.length;

3.数组的定义

关于数组的定义,其实有很多种定义方法,接下来,我会给你们介绍几种定义数组的方法。

//第一种定义数组的方法:
int[] array1 = {1,2,3};//直接赋值(静态初始化)
                      //int[]是数组的类型,array为数组名,随意取名字

//第二种定义数组的方法:
int[] array2 = new int[]{1,2,3,4};//数组的动态初始化


//第三种定义数组的方法:
int[] array3 = new int[10];//只是分配了内存,但是没有进行赋值,默认值都是0,

//第四种定义数组的方法:
int[] array4;
array = new int[]{1,2,3};//一定要为数组符初值,不然编译器会报错,


对于没有赋初值的数组,编译器会自动赋值,以下表格是不同类型的数组,初值大小:

类型初始值
byte0
short0
int0
long0
float0.0f
double0.0
char‘/u0000’
booleanfalse

4.注意事项

  • 静态初始化虽然没有指定数组的长度,编译器在编译时会根据{}中元素个数来确定数组的长度。
  • 静态初始化时, {}中数据类型必须与[]前数据类型一致。
  • 静态初始化可以简写,省去后面的new T[]。T可以为任意数据类型。
  • 如果数组中存储元素类型为引用类型,默认值为null。

二、数组的使用

1.对数组中的元素进行访问

数组在内存中是一段连续的空间,空间的编号都是从0开始的,依次递增,该编号称为数组的下标,数组可以通过
索引访问其任意位置的元素。

int[] array = {1,2,3,4,5,6,7,8};
System.out.println(array[0]);//1
System.out.println(array[1]);//2
System.out.println(array[2]);//3
System.out.println(array[3]);//4
System.out.println(array[4]);//5

//当然也可以对数组中的元素进行修改,、
array[0]=111;
System.out.println(array[0]);//111

2.注意事项

(1)数组是一段连续的内存空间,因此支持随机访问,即通过下标访问快速访问数组中任意位置的元素。

(2)下标从0开始,介于[0, N)之间不包含N,N为元素个数,不能越界,否则会报出下标越界异常。

在这里插入图片描述

所以在访问数组元素时一定不要越界!!!

3.对数组的遍历

对于什么是对数组的遍历,就是将数组的元素全都访问一遍,如将数组中的全部元素的打印一遍。

第一种数组遍历方法:直接打印数组中的每一个元素

Demo:

int[] array = {1,2,3,4};
System.out.println(array[0]);
System.out.println(array[1]);
System.out.println(array[2]);
System.out.println(array[3]);

本方法对于元素个数较少的数组来说可以应用,但对于有几十个元素的数组来说,太麻烦了,接下来为大家讲解第二种方法。

第二种数组遍历的方法:采用for循环打印

Demo:

int[] array = {1,2,3,4};
for(int i = 0;i < array.length;i++) {
    System.out.print(array[i]);
}

第三种数组遍历的方法:采用foreach的方式

Demo:

int[] array = {1,2,3,4,5};
for(int x : array) {
    System.out.print(x+" ");
}

foreach的语法格式:冒号右边写数组名,左边写由数组当中数据类型定义的变量。

但是foreach也有局限性,如果要求去访问数组中的某一个元素的时候,就不能进行访问,显然这时的for循环就占据了优势。

第四种遍历数组的方法:采用专门的工具

Demo:

import java.util.Arrays;

int[] array = {1,2,3,4,5};
String ret = Arrays.toString(array);//将数组转换成字符串,然后返回.
System.out.println(ret);

注:要使用Arrays这个工具时要导入其专有的包!

三、数组在内存的存在形式

1.数组是一种引用变量

基本数据类型创建的变量,称为基本变量,该变量空间中直接存放的是其所对应的值;
而引用数据类型创建的变量,一般称为对象的引用,其空间中存储的是对象所在空间的地址,数组就是一种引用数据类型

int a = 10;
int b = 20;
int[] array =new int[]{1,2,3,4};

在上述代码中,a、b、arr,都是方法内部的变量,因此其空间都在main方法对应的栈帧中分配。
a、b是内置类型的变量,因此其空间中保存的就是给该变量初始化的值。
array是数组类型的引用变量,其内部保存的内容可以简单理解成是数组在堆空间中的首地址。

在这里插入图片描述

从图中可以看出,引用变量与基本数据类型的变量的不同,引用变量不会存储对象本身,可以简单理解成存储的是对象在堆中空间的起始地址。通过该地址,引用变量便可以去操作对象,(但其实这“地址”是经过哈希得到的),此前不要求掌握该知识点。

2了解引用变量(数组)

Demo:

int[] array1 = {1,2,3,4};
System.out.print(Arrays.toString(array1));//打印[1,2,3,4]

int[] array2 = array1;
array2[1] = 99;
System.out.print(Arrays.toString(array1));//打印[1,99,3,4]
System.out.print(Arrays.toString(array2));//打印[1,99,3,4]

代码的解析:

在这里插入图片描述

首先创建一个数组array1,并初始化赋值为1,2,3,4,然后打印数组array1,接着申请另一款空间,用来创建array2,且array2=array1,说明两个数组都指向同一块空间,修改array2中的第二个元素也就相当于修改了array1中对应的元素。

3.注意事项

(1)一个引用不能同时指向多个对象
例:
在这里插入图片描述

(2)一个对象可以被多个引用所指向
例:
在这里插入图片描述

(3)

int[] array = 0;
/*这种写法是错误的,因为0是一种基本数据类型,
 *而array是一种引用数据类型,如果将0改成null就不会错。
 *即代表这个引用不指向任何对象
*/

注:null在Java中表示空引用,也就是一个不指向对象的引用,并不是0的意思。

四、数组的应用场景

1.用来保存数据

Demo:

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

2.数组作为方法的参数

Demo:

public class TestDemo {
    public static void func1 (int[] array1) {
        array = new int[10];
    }
    public static void func2 (int[] array) {
        array[0] = 99;
    }
    public static void main(String[] args) {
        int[] array = {1,2,3,4};
        func1(array);
        System.out.println(Arrays.toString(array));
        func2(array);
        System.out.println(Arrays.toString(array));

    }
}

代码解释:对func1方法进行解释就行,func2同理

在这里插入图片描述

形参的指向改变不会影响实参的指向改变!!!

3.数组作为方法的返回值

Demo:

public class TestDemo {
    public static int[] func() {
        int[] array = {1,2,3,4};
        return array;
    }
    public static void main(String[] args) {
        int[] ret = func();
        System.out.println(Arrays.toString(ret));
    }
}

五、数组的练习

小伙伴们,也可以直接去写一下,再过来看示例代码,增强自己的代码能力。

1.数组转成字符串

Demo:

//第一种是利用编译器自带的工具
int[] array = {1,2,3};
String ret = Arrays.toString(array);
System.out,print(ret);


//第二种利用自己写的方法去解决
public class TestDemo {
    public static String mytoString(int[] arr) {
        if(arr == null){
            return "null";
        }
        String ret = "[";
        for(int i=0;i<arr.length;i++) {
            ret+=arr[i];
            if(i!=arr.length-1) {
                ret+=",";
            }
        }
        ret+="]";
        return ret;
    }
    public static void main(String[] args) {
        int[] array = {1,2,3,4,5,6};
        String ret = mytoString(array);
        System.out.println(ret);
    }
    
    
//打印结果为 [1,2,3,4,5,6]

2.数组的拷贝

本练习还是采用两种方式,一种是编译器自带的,另一种是自己所写的方法。

下面是错误代码!!!

/*
        int[] array1 = {1,2,3};
        int[] array2 = array1;
*/
//这不是对数组的拷贝,因为没有产生新的空间

Demo:

//第一种:自己写的方法
public class TestDemo {
    public static void main(String[] args) {
        int[] array1 = {1,2,3,4};
        int[] array2 = new int[array1.length];

        for(int i=0;i<array1.length;i++) {
            array2[i]=array1[i];
        }

        System.out.println(Arrays.toString(array1));
        System.out.println("复制之后的数组为:");
        System.out.println(Arrays.toString(array2));

    }
}

//最后输出为:
  [1,2,3,4]
  复制之后的数组为:
  [1,2,3,4] 
    
    
//第二种编译器自带的方法
    import java.util.Arrays;
    
    int[] array1 = {1,2,3,4};
    int[] array2 = Arrays.copyOf(array1,array1.length);

    System.out.println(Arrays.toString(array1));
    System.out.println(Arrays.toString(array2));

//最后输出为:
    [1,2,3,4] 
    [1,2,3,4] 

显然自己写的方法没有编译器自带的工具简洁,但是不能说有了这些工具就可以不用写代码了,自己的能力还是要加强,这些工具只是帮我们减轻了负担,但是原理还是要自己弄懂。

3.求数组的平均值

由于这道题比较简单,本博主直接就给答案咯。

Demo:

public class TestDemo {
    public static void main(String[] args) {
        int[] array = {23,45,2,74,86,24};
        int sum = 0 ;
        double ave = 0.0f;
        for(int i=0; i<array.length;i++) {
             sum=sum + array[i];
        }
         ave = sum/array.length;
        System.out.println(ave);

    }
}

//最后结果:
  42.0

4.数组排序(冒泡排序)

算法思路:

  • 将数组中相邻元素从前往后依次进行比较,如果前一个元素比后一个元素大,则交换,一趟下来后最大元素
    就在数组的末尾
  • 依次从上上述过程,直到数组中所有的元素都排列好

Demo:

public class TestDemo {
    public static void main(String[] args) {
        int[] array = {23,4,53,1,765,52,356,9,7,2,324};
        System.out.println("排序之前的数组:");
        System.out.println(Arrays.toString(array));
        for(int j = 1;j<array.length;j++) {
            for(int i = 0;i<array.length-j;i++) {
                if(array[i]>array[i+1]) {
                    int tmp = array[i];    //交换数据
                    array[i] = array[i+1];
                    array[i+1] = tmp;
                }
            }
        }
        System.out.println("排序之后的数组:");
        System.out.println(Arrays.toString(array));
    }
}


//排序之前的数组:
[23, 4, 53, 1, 765, 52, 356, 9, 7, 2, 324]
//排序之后的数组:
[1, 2, 4, 7, 9, 23, 52, 53, 324, 356, 765]

5.数组逆序

题目理解:就是给定一个数组,将其倒序输出。

思路:
先定义两个下标分别指向数组的第一个元素和最后一个元素,将数组中的第一个元素和最后一个元素交换位置,然后指向第一个元素的下标进行加加,指向最后一个元素的下标进行减减。

Demo:

public class TestDemo {
    public static void main(String[] args) {
        int[] array = {1,2,3,4,5,6};
        int left = 0;
        int right = array.length-1;
        System.out.println("排序之前的数组:");
        System.out.println(Arrays.toString(array));

        while(left<=right) {
            int tmp = array[left];
            array[left] = array[right];
            array[right] = tmp;
            left++;
            right--;
        }
        System.out.println("排序之后的数组:");
        System.out.println(Arrays.toString(array));
    }
}

//运行结果:
排序之前的数组:
[1, 2, 3, 4, 5, 6]
排序之后的数组:
[6, 5, 4, 3, 2, 1]

六、二维数组

1.二维数组的定义

关于二维数组的定义有很多种方法,这里咱们介绍最常用的三种。

Demo:

//第一种定义二维数组的方法:(行数和列数都知道的情况)
//  数组类型[][] 数组名 = new 数组类型[行数][列数];

 int[][] arr = new int[2][3];//定义一个3行4列的数组

在这里插入图片描述

//第二种定义数组的方法:(只知道行数,不知道列数)
//  数组类型[][] 数组名 = new 数组类型[行数][];

int[][] arr = new int[2][];

在这里插入图片描述

//第三种定义二维数组的方法:(知道具体的数据)
//  数据类型[][] 数组名 = {{第0行初始值},{第1行初始值},···,{第n行初始值}};

int[][] arr = {{1,2},{3,4,5},{6,7,8,9}};

在这里插入图片描述

以上就是关于二维数组最常见的定义方法!!!

2.二维数组的打印

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

Demo:

public class TestDemo {
    public static void main(String[] args) {
        int[][] array = {{1,2},{3,4,5},{6,7,8,9}};
        for(int i=0;i< array.length;i++) {
            for(int j=0;j<array[i].length;j++) {
                System.out.print(array[i][j]+" ");
            }
            System.out.println(" ");
        }
    }
}

//运行结果:
1 2  
3 4 5  
6 7 8 9 

由于二维数组的应用和一维数组并没有太大的区别,所以,在这里不做介绍,学会如何定义二维数组就好。

  • 47
    点赞
  • 132
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 15
    评论
树状数组(Fenwick Tree)是一种用于快速维护数组前缀和的数据结构。它可以在 $O(\log n)$ 的时间内完成单点修改和前缀查询操作,比线段树更加简洁高效。 下面是 Java 实现的树状数组详解: 首先,在 Java 我们需要使用数组来表示树状数组,如下: ``` int[] tree; ``` 接着,我们需要实现两个基本操作:单点修改和前缀查询。 单点修改的实现如下: ``` void update(int index, int value) { while (index < tree.length) { tree[index] += value; index += index & -index; } } ``` 该函数的参数 `index` 表示要修改的位置,`value` 表示修改的值。在函数内部,我们使用了一个 `while` 循环不断向上更新树状数组相应的节点,直到到达根节点为止。具体来说,我们首先将 `tree[index]` 加上 `value`,然后将 `index` 加上其最后一位为 1 的二进制数,这样就可以更新其父节点了。例如,当 `index` 为 6 时,其二进制表示为 110,最后一位为 2^1,加上后变为 111,即 7,这样就可以更新节点 7 了。 前缀查询的实现如下: ``` int query(int index) { int sum = 0; while (index > 0) { sum += tree[index]; index -= index & -index; } return sum; } ``` 该函数的参数 `index` 表示要查询的前缀的结束位置,即查询 $[1, index]$ 的和。在函数内部,我们同样使用了一个 `while` 循环不断向前查询树状数组相应的节点,直到到达 0 为止。具体来说,我们首先将 `sum` 加上 `tree[index]`,然后将 `index` 减去其最后一位为 1 的二进制数,这样就可以查询其前一个节点了。例如,当 `index` 为 6 时,其二进制表示为 110,最后一位为 2^1,减去后变为 100,即 4,这样就可以查询节点 4 的值了。 最后,我们还需要初始化树状数组,将其全部置为 0。初始化的实现如下: ``` void init(int[] nums) { tree = new int[nums.length + 1]; for (int i = 1; i <= nums.length; i++) { update(i, nums[i - 1]); } } ``` 该函数的参数 `nums` 表示初始数组的值。在函数内部,我们首先创建一个长度为 `nums.length + 1` 的数组 `tree`,然后逐个将 `nums` 的元素插入到树状数组。具体来说,我们调用 `update(i, nums[i - 1])` 来将 `nums[i - 1]` 插入到树状数组的第 `i` 个位置。 到此为止,我们就完成了树状数组的实现。可以看到,树状数组的代码比线段树要简洁很多,而且效率也更高。
评论 15
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

要努力点

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值