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]);//1System.out.println(array[1]);//2System.out.println(array[2]);//3System.out.println(array[3]);//4System.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 23 4 56 7 8 9 

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