0


【JavaSE系列】第六话 —— 数组

前言

** 从数组这一章开始,Java就开始逐渐的和C语言有点区别了。**

** 因此,大家一定要端起自己的小板凳,认认真真的看完接下来的博客,希望看完博客的都是干货满满哦。**


一、思维导图


二、数组的基本概念

2.1 为什么要使用数组

** 假如说,我们有这样的一个需求,需要存储10个整型变量的值;**

** 那么,按照以前的所学知识,我们会定义10个变量;但是,定义10个变量 的数量有一点点多,而且 它们全部都是整型变量。**

** 为了能够满足 可以存储多种相同类型的变量,于是 便规定了一种新的数据类型:数组。**

2.2 什么是数组

数组:可以看成是相同类型元素的一个集合,在内存中是一段连续的空间。

【注意】

  1. 数组中存放的元素其类型相同;
  2. 数组的空间是连在一起的 ;
  3. 每个空间有自己的编号,即数组的下标(从0开始编号,我们可以运用下标 来对数组中的元素进行访问和修改)。

2.3 数组的创建及初始化

2.3.1 数组的定义

** 在Java中,数组和C语言中的是有着区别的;**

** 在Java中,int [ ] 是数组的 数据类型,那么我们可以这样来定义数组:**

** int [ ] array —> int [ ] 是数据类型,array是 变量。**

(1)第一种方式:

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

** (2)第二种方式:**

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

** 【注意】在前两种定义和初始化的过程中,中括号[ ]当中 不能有任何的数字,否则就会报错;当然,也不必当心编译器会不知道 所定义的数组的长度,编译器会自动推导出来的。**

我们可以通过 数组名.length自动获取当前数组的长度:

** (3)定义的第三种方式**

int[] array3 = new int[10];

** 这一种方式的主要区别与前面两种的是:没有初始化。**

** 那么我们可以来测试一下 有没有默认值:**

** 【说明】三种定义方式 可以根据自己的需求,来决定到底用哪一种。但是,平常我们用的最多的是第一种方式(第一种方式 和 第二种方式 其实是一模一样的,但是它更加简便)。**

** 第一种方式 和 第二种方式 中括号[ ] 里面 是肯定不可以有数字的,**

** 至于第三种定义方式,肯定是要有数字的,否则它肯定推不了数组的长度(后面都没有赋值)。**

【注意】

** 如果没有对数组进行初始化,数组中元素有其默认值:**

** 如果数组中存储元素类型为基类类型,默认值为基类类型对应的默认值:**

** 如果数组中存储元素类型为引用类型,默认值为null 。**

数组越界问题:

2.4 数组的使用

2.4.1 数组中元素访问

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

** 当然, 访问数组下标的时候,千万不能越界:**

2.4.2 数组遍历

** "遍历" **是指将数组中的所有元素都访问一遍,

** 访问是指对数组中的元素进行某种操作(比如说:打印)。**

第一种打印方式(for 循环):

第二种打印方式(for-each):

**【注意】冒号 左边和右边的类型要相匹配。 **

** 普通的for循环和增强for循环的区别:**

** 普通for循环 可以拿到数组的下标,通过下标对数组进行操作;**

** 增强for循环 我们只是拿到了变量的值,与下标无关。**

** 第三种打印方式(借用Java本身提供的一些方法来实现数组的打印):**

** 这里首先需要介绍一个 工具类(可以理解为C语言里面的头文件):Arrays**

** 其主要作用是:帮助对数组进行一个操作(详情可查找帮助手册)。**


三、数组是引用类型

3.1 初始JVM的内存分布

** 划分内存的好处是:简洁、方便管理。就比如是这样:**

** 而现在我们只简单关心堆 和 虚拟机栈这两块空间,后序JVM中还会更详细介绍。 **

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

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

举个例子:

public class TestDemo {
    public static void main(String[] args) {
        int a = 10;//a是局部变量,我们把它放到了Java虚拟机栈 当中
    }
}

我们来看一下定义数组内存分配的情况:

    public static void main(String[] args) {
        
        int[] array = {1,2,3};
        System.out.println(array);
    }

【注意】引用是一个变量,里面存的是地址。

3.2 认识null

** null 在 Java 中表示 "空引用" , 即:一个不指向对象的引用。**

3.3 分析引用变量

分析一:

    public static void main(String[] args) {
        int[] array1 = {1,2,3,4};
        int[] array2 = array1;
        System.out.println("array1:"+Arrays.toString(array1));
        System.out.println("array2:"+Arrays.toString(array2));
    }

** 此时,如果 array2的对象 发生了改变,那么array2的对象也必然会发生改变:**

** 所以可以得出一个结论:不要说array2引用指向了引用,而是说 这个引用array2 指向了 array1所指向的对象。**

分析二:

3.4 四个小问题

问题一:引用能指向引用吗?

答案:不能,这句说法就是有问题的(引用只能指向对象,引用不能指向引用)。

**array1 = array2 代表:array1这个引用 引用了array2所引用的对象。 **

问题二:一个引用可以同时指向多个对象吗?

答案:不能,可以不同时指向多个对象,但只能同时指向一个对象(指向另外一个对象的时候,上一次所存储的对象的地址被销毁掉)。

问题三:引用一定是在栈上吗?

**答案:不是的,现在之所以把它画在栈上,是因为我们目前只接触到了 局部变量,还没有接触到成员变量;这个成员变量在后面会继续介绍的。 **

问题四:引用赋值null代表啥?

**答案:int[ ] array = null; 代表整个引用不指向任何对象。 **


四、数组的使用方式

4.1 保存数据

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

4.2 作为函数的参数

** 首先我们需要一个不带数组的简单类型的方法: **

一步一步的来分析:

** 首先,在栈上分配了一块内存空间存储 x=10;**

** 然后,进入func1方法,在栈上又开辟了一块int[ ] 类型的内存,把10赋给a;**

** 接着,又把20赋值给了a;**

** 当最终func1方法结束以后,x的值仍然还是10,并没有发生改变;所以打印的还是10,并没有打印20。**

接下来我们再来看一下引用类型的方法:

**第一种: **

第二种:

**【说明】只要new一下,就会开辟一块新的空间,然后数组就会存储一块新的地址。 **

4.3 作为函数的返回值

** C语言里面是不能返回数组的;**

** 但是Java里面可以返回数组,不过返回的都是地址。**

【注意】函数回收 只会把 栈上面的内存回收掉。


五、数组练习

5.1 写一个函数myToString,将数组以字符串的形式输出

import java.util.Arrays;

public class TestDemo {
    public static String myToString(int[] array) {
        String str = "[";
        for (int i = 0; i < array.length; i++) {
            str = str + array[i];
            if(i!=array.length-1){
                str+=",";
            }
        }
        str = str + "]";
        return str;
    }
    public static void main(String[] args) {
        int[] array = {1,2,3,4};
        String ret = myToString(array);
        System.out.println(ret);
    }
}

5.2 数组拷贝

第一种拷贝方式:

** 使用for循环进行拷贝:**

import java.util.Arrays;

public class TestDemo {
    //数组拷贝
    public static void main(String[] args) {
        int[] array = {1,2,3,4};

        int[] copy = new int[array.length];

        for (int i = 0; i < array.length; i++) {
            copy[i] = array[i];
        }
        System.out.println(Arrays.toString(copy));
    }

}

第二种拷贝方式:

** 通过Arrays这个类自带的拷贝方法(Arrays.copyOf):**

import java.util.Arrays;

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

这个方法虽然说是拷贝,但也可以理解成扩容:

第三种拷贝方式:

System.arraycopy:

** 【说明】src是你要拷贝的数组,srcPos是开始拷贝的下标,dest是目的地数组,destPos是从目的地数组的位置开始拷贝,length是你要拷贝的长度。**

至于要取什么值,可以自己去试试看,确实是很有意思的:

import java.util.Arrays;

public class TestDemo {
    public static void main(String[] args) {
        int[] array = {1,2,3,4};
        int[] copy = new int[array.length];
        System.arraycopy(array,0,copy,0,array.length);

        System.out.println(Arrays.toString(copy));
    }
}

第四种拷贝方式:

Arrays.copyOf(这个也是以后用的非常多的拷贝方式):

import java.util.Arrays;

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

怎么去证明这个:

5.3 求一个整型数组的平均值

import java.util.Arrays;

public class TestDemo {
    public static double avg(int[] array){
        int sum = 0;
        for (int i = 0; i < array.length; i++) {
            sum = sum+array[i];
        }
        return sum*1.0/array.length;
    }
    public static void main(String[] args) {
        int[] array = {1,2,3,4,5};
        System.out.println(avg(array));
    }
}

5.4 查找数组中指定元素(顺序查找)

顺序查找

根据给定的数组,然后去找数组中的某一个值;

需要从头开始,一个一个的去往下找,看 哪一个是所需要找的值,找到就可以返回它的下标。

import java.util.Arrays;

public class TestDemo {
    public static int search(int[] array,int key){
        for (int i = 0; i < array.length; i++) {
            if(array[i]==key){
                return i;
            }
        }
        //如果代码走到这里,说明退出了for循环,或者是没有进入for循环
        return -1;//返回-1的原因是 因为数组没有负数下标(当然随便一个负数都可以)
    }
    public static void main(String[] args) {
        int[] array = {1,2,3,4,5};
        int index = search(array,4);
        if (index == -1){
            System.out.println("没有你要找的关键字!");
        }else {
            System.out.println("找到了你要的关键字,下标是:"+index);
        }
    }

}

**最坏的情况下,需要遍历所有数组的元素一个一个的去找,即时间复杂度是O(N),这个效率就会比较慢。 **

5.5 查找数组中指定元素(二分查找)

由于顺序查找的效率可能比较慢,下面来介绍一种比较高效的查找方法:二分查找。

二分查找

二分查找需要一个前提条件:要查找的这一数组中的数据必须是有序的。

import java.util.Arrays;

public class TestDemo {
    public static int binarySearch(int[] array,int key){
        int left = 0;
        int right = array.length-1;
        while (left <= right){
            int mid = (left + right)/2;
            if (array[mid] < key){
                left = mid + 1;
            }else if (array[mid]==key){
                return mid;
            }else {
                right = mid - 1;
            }
        }
        //假如代码可以走到这里,说明 left>right
        return -1;
    }
    public static void main(String[] args) {
        int[] array = {1,2,3,4,5};
        int index = search(array,4);
        if (index == -1){
            System.out.println("没有你要找的关键字!");
        }else {
            System.out.println("找到了你要的关键字,下标是:"+index);
        }
    }
}

** 那么有的人就会说,二分查找限制好大啊,还需要 有序数组;**

** 可以先对数组进行排序,然后再去查找某个元素;**

** 其实,Java是很友好的,它提供了一个专门去排序的sort方法(在工具类Arrays里面):**

Arrays.sort(要排序的数组名);

【说明】如果面试的时候,一定要先和面试官说一下是否可以先用Arrays.sout排序,要不然面试官是想让你好好找元素,而不是要来排序的;如果在不排序的情况下,再把问题解决了那才是最厉害的。

【注意】

  1. sort方法是没有返回值的,并且默认是升序;
  2. 对于现在的简单数据类型,能不能用降序排序 ——> 答案是做不到;
  3. 怎么样可以做到 指定的升序或者降序,我们需要继续向后学习,去学习接口;
  4. 只有所要排序的数据 是引用类型的时候 才可以做到 指定的升序和降序。至于具体的做法,则会在抽象类和接口的那一部分会介绍的。

【说明】 其实在Java中也已经有排序的方法了,甚至于以后可以不用自己写排序了(binarySearch方法):

5.6 冒泡排序

算法思路:

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

代码示例:

import java.util.Arrays;

public class TestDemo {
    public static void bubbleSort(int[] array){
        //[0~length-1) i代表的是要比较的趟数
        for (int i = 0; i < array.length-1; i++) {
            boolean flg = false;

            //在这里可以不用减i,如果减了i说明进行了一次优化
            for (int j = 0; j < array.length-1-i; j++) {
                if (array[j] > array[j+1]){
                    int tmp = array[j];
                    array[j] = array[j+1];
                    array[j+1] = tmp;
                    flg = true;
                }
            }
            if (flg == false){
                break;
            }
        }
    }
    public static void main(String[] args) {
        int[] array = {12,5,7,3,8};
        System.out.println("排序之前:"+Arrays.toString(array));
        bubbleSort(array);
        System.out.println("排序之后:"+Arrays.toString(array));
    }

** 图示:**

**最后的优化: **

** 检查是否发生了交换,如果没有就说明已经有序了,就不需要再进行接下来的交换了。**

** 即上面所示例的代码 flg那部分是false、true、false或者是true、false、true都可以,亲测有效哦!**

** 冒泡排序的思想暂时就介绍到这里了,上面的代码示例也是稍微优化过的代码;**

** 如果有时间的话,会专门出一期关于排序的博客,会从头到尾的介绍一下优化过程之类的......**

5.7 数组逆序

思路:

** 设定两个下标, 分别指向第一个元素和最后一个元素,交换两个位置的元素;**

然后让前一个下标自增,后一个下标自减,循环继续即可。

代码示例:

import java.util.Arrays;

public class TestDemo {
    public static void reserve(int[] array){
        int left = 0;
        int right = array.length-1;
        int tmp = 0;
        while (left<right){
            tmp = array[left];
            array[left] = array[right];
            array[right] = tmp;
            left++;
            right--;
        }
    }
    public static void main(String[] args) {
        int[] array = {11,22,33,44,55,66,77,88,99,1010};
        System.out.println("逆置之前:"+ Arrays.toString(array));
        reserve(array);
        System.out.println("逆置之后:"+Arrays.toString(array));
    }
}

5.8 其他

** 浅浅的介绍一下 Arrays这个工具类,里面具有很多的方法,详情可以去看看帮助手册:**

** 通过之前的介绍,我们已经知道了 打印数组的toSpring()方法,以及 拷贝数组的copyOf()方法,这里就不过多的介绍了嗷。**

** 现在来介绍一下其他的方法:**

** copyOfRange()方法 ——> 这个只是拷贝一部分的情况:**


六、二维数组

**【说明】这个二维数组也会和C语言中的二维数组有细微的差别,请听我娓娓道来: **

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

基本语法:

定义及初始化:

第一种(以直接赋值的方式):

 int[][] array1 = {{1,2,3},{4,5,6}};

第二种(直接new):

int[][] array2 = new int[][] {{1,2,3},{4,5,6}};

** 前面两种方式(后面有自己 赋初值的) 都不能在 中括号里面 写数字,否则就会报错;它会自动识别并给出相应的数据。**

第三种(没有赋初始值的):

    int[][] array3 = new int[2][3];

** 此时右端需要数值(不然判断不出来有几行几列),并且默认初始值都是0。**

打印二维数组:

** 可是,难道需要每一次都去改变 i 和 j 的值来改变 行和列的嘛,这明显不现实;**

** 因此咱们还需要修改一下:**

具体分析一下:

用各种方式打印二维数组的代码示例:

import java.util.Arrays;

public class TestDemo {
    public static void main(String[] args) {
        int[][] array1 = {{1,2,3},{4,5,6}};
        for (int i = 0; i < array1.length; i++) {
            for (int j = 0; j < array1[i].length; j++) {
                System.out.print(array1[i][j] + " ");
            }
            System.out.println();
        }
        System.out.println("使用foreach来进行打印:");
        for (int[] tmp:array1){
            for (int x:tmp) {
                System.out.print(x + " ");
            }
            System.out.println();
        }
        System.out.println("使用toString方法来进行打印:");
        System.out.println(Arrays.deepToString(array1));
    }

}

** 上面示例的结果: **


总结

这一话的需要知道的内容就这么多了,

如果有啥不到位的地方欢迎指出来,大家互相督促、共同进步啊。

当然啦如果铁铁们可以一键三连那就更棒了,特别特别感谢 ୧(๑•̀⌄•́๑)૭ 。


本文转载自: https://blog.csdn.net/qq_53362595/article/details/123646003
版权归原作者 哎呀是小张啊 所有, 如有侵权,请联系我们删除。

“【JavaSE系列】第六话 &mdash;&mdash; 数组”的评论:

还没有评论