javaSE - Arrays - 数组的定义与使用

一、数组基本用法

1.1、什么是数组

数组本质上就是让我们能 “批量” 创建相同类型的变量
也可以说是存储一组相同数据类型的数据的集合

如:
如果需要表示两个数据, 那么直接创建两个变量即可 int a; int b
如果需要表示五个数据, 那么可以创建五个变量 int a1; int a2; int a3; int a4; int a5;
但是如果需要表示一万个数据, 那么就不能创建一万个变量了. 这时候就需要使用数组, 帮我们批量创建

注意事项: 在 Java 中, 数组中包含的变量必须是 相同类型

1.2、创建数组

基本语法
创建数组的三种方式:

// 动态初始化
数据类型[] 数组名称 = new 数据类型 [] { 初始化数据 };
// 静态初始化
数据类型[] 数组名称 = { 初始化数据 };

类型[] 数组名 = new 类型[元素个数];

代码示例:

public static void main(String[] args) {
        //动态初始化,数组里面存放了 1 2 3 4 
        int[] array2 = new int[]{1, 2, 3, 4};
        //静态初始化,数组里面存放了 1 2 3 4
        int[] array1 = {1, 2, 3, 4};
        //这里只是没有初始化数组的内容,默认里面存放的是5个0
        int[] array3 = new int[5];
    }

在这里插入图片描述
而且对数组有一定了解的人,都知道数组的每个元素都有一个下标(从0开始),方便去寻找寻找元素.
在这里插入图片描述

1.3、总结

一套讲解下来,你会发现在Java中 是这么来表达一个数组:int[] array
其实数组也可以写成
int arr[] = {1, 2, 3};
和 C 语言一样. 但是我们还是更推荐写成 int[] arr 的形式. int和 [] 是一个整体.,因此,其实在Java中数组的写法更为准确。

但是不能像C语言一样这样写 int array[10] = {0};
我们前面也看到了,在创建一个数组时,[ ]里是不能有具体数字的存在,除了第三种方法,其它的,一律不行

1.4、数组的使用

1.4.1、代码示例: 获取长度 , 访问元素

public static void main(String[] args) {
        int[] array = new int[]{1,2,3,4,5,6};
        //计算数组的长度
        int len = array.length;
        System.out.println("数组的长度为:" + len);
        //访问数组的元素
        System.out.println(array[1]);
        System.out.println(array[2]);
        System.out.println(array[3]);

    }

在这里插入图片描述

1.4.2、代码示例: 下标越界

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

在这里插入图片描述
抛出了 java.lang.ArrayIndexOutOfBoundsException 异常. 使用数组一定要下标谨防越界.

1.4.3、码示例: 遍历数组

所谓 “遍历” 是指将数组中的所有元素都访问一遍, 不重不漏. 通常需要搭配循环语句

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

在这里插入图片描述

1.1.4、代码示例: 使用 for-each 遍历数组

public static void main(String[] args) {
        int[] array = {1, 2, 3, 4, 5};
        for(定义一个 与数组元素类型 相同的变量 : 数组名)
 
  什么意思呢?
for-each(增强for循环), 数组名部分,表示的意思 遍历访问数组的元素
将访问的元素赋给 冒号前面 定义的 与数组元素类型相同 的变量
 我们只需要 将该变量每次得到的元素值,打印
  就能做到不依靠元素下标,遍历打印数组所有元素

        for (int x: array) {
            System.out.print(x + " ");
        }
    }

在这里插入图片描述

那么 for 和 foreach 两者有什么区别?
最大的区别在于,for是可以拿到元素下标,而foreach拿不到元素下标
for循环用到的地方很多,但是foreach呢?
当我们只需要元素的值时,就是使用foreach,
当我们还需要元素的下标时,就用for。
for-each 是 for 循环的另外一种使用方式. 能够更方便的完成对数组的遍历. 可以避免循环条件和更新语句写错

注意事项

  1. 使用 arr.length 能够获取到数组的长度. . 这个操作为成员访问操作符. 后面在面向对象中会经常用到.
  2. 使用 [ ] 按下标取数组元素. 需要注意, 下标从 0 开始计数
  3. 使用 [ ] 操作既能读取数据, 也能修改数据.
  4. 下标访问操作不能超出有效范围 [0, length - 1] , 如果超出有效范围, 会出现下标越界异常

代码实例3(借助Java的操作数组的工具类 Arrays)

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

在这里插入图片描述
Arrays工具类还有很多的方法:Arrays工具类的常用方法

二、数组作为方法的参数

首先要说一下 jvm内存模型

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

在这里插入图片描述

public static void main(String[] args) {
        int[] array = null;
        System.out.println(array.length);
    }

在这里插入图片描述

2.1、基本用法

2.1.1、代码示例: 打印数组内容

/**
     * 打印数组的内容
     * @param array 数组名
     */
    public static void printf(int[] array){
        for (int i = 0; i < array.length; i++) {
            System.out.print(array[i] + " ");
        }
    }
    public static void main(String[] args) {
        int[] array = {1, 2, 3, 4, 5, 6};
        printf(array);
    }

在这里插入图片描述

2.2.2、通过下面题目更好的理解引用类型

下面两题的输出结果是什么?
题目1:

public static void main(String[] args) {
        int[] array = {1,2,3,4,5};
        func1(array);
        System.out.println(Arrays.toString(array));// 图 18

    }
    public static void func1(int[] array){
        array = new int[]{11,2,13,4,51};
    }

题目2:

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

在这里插入图片描述

2.2.3、一个引用 是否 能同时 指向 多个对象?

public static void main(String[] args) {
        int[] array1 = new int[]{1,2,3,4,5};
        array1 = new int[10];
        array1 = new int[2];
        array1 = new int[4];
    }

答案是不能,如果前面认真看了,就该知道此时的array1,存储的地址,已经被改变(array是一个局部变量,意味着存的值是可以被改变的),现在存的是 new int[4] 的这个对象的地址, 而不是说,存几个对象的地址。

一个引用只能指向一个对象(一个引用只能保存一个对象的地址)

2.2.4、引用 就一定在栈上吗?

答案是不一定的,因为一个变量在不在栈上,是你变量的性质决定的
如果你的引用是一个局部变量,那就一定在栈上 实例成员变量那就不一定了。(先告诉你们有这个概念,等后面讲到成员变量时再说)

局部变量的引用保存在栈上, new 出的对象保存在堆上.

堆的空间非常大, 栈的空间比较小.    因为 堆 是整个 JVM 共享一个,
而 栈 每个线程具有一份 (一个 Java 程序中可能存在多个栈)

三、数组作为方法的返回值

3.1、代码示例: 写一个方法, 将数组中的每个元素都 * 2(直接修改原数组)

public static void main(String[] args) {
        //代码示例: 写一个方法, 将数组中的每个元素都 * 2
        // 直接修改原数组
        int[] array = {1,2,3,4,5};
        mul2(array);
        System.out.println(Arrays.toString(array));
    }
    public static void mul2(int[] array){
        for(int i = 0; i < array.length; i++){
            array[i] = array[i] * 2;
        }
    }

在这里插入图片描述

这个代码固然可行, 但是破坏了原有数组. 有时候我们不希望破坏原数组, 就需要在方法内部创建一个新的数组, 并由方法返回出来

3.2、代码示例: 写一个方法, 将数组中的每个元素都 * 2(将原数组拷贝一份,改变拷贝的数组,并返回拷贝的数组)

public static void main(String[] args) {
        //代码示例: 写一个方法, 将数组中的每个元素都 * 2
        // 直接修改原数组
        int[] array = {1,2,3,4,5};
        int[] ret = mul2(array);
        System.out.println("原来的数组:" + Arrays.toString(array));
        System.out.println("改变后的数组:" + Arrays.toString(ret));
    }

    public static int[] mul2(int[] array){
        int[] str = new int[array.length];
        for(int i = 0; i < array.length; i++){
            str[i] = array[i] * 2;
        }
        return str;
    }

在这里插入图片描述
这样的话就不会破坏原有数组了.
另外由于数组是引用类型, 返回的时候只是将这个数组的首地址返回给函数调用者, 没有拷贝数组内容, 从而比较高效

四、数组练习

4.1、模拟实现 ToString方法 ,数组转字符串

这是java自带的ToString方法

在这里插入图片描述
模拟实现:

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

4.2、数组拷贝 代码示例 拷贝整个数组

拷贝整个数组

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

在这里插入图片描述

模拟实现mycopyDf

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

    public static int[] mycopyDf(int[] array, int len){
        int[] str = new int[len];
        for (int i = 0; i < len; i++){
            str[i] = array[i];
        }
        return str;
    }

4.3、数组拷贝 代码示例 拷贝某个范围

拷贝某个范围

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

在这里插入图片描述

4.4、找数组中的最大元素

给定一个整型数组, 找到其中的最大元素 (找最小元素同理)

public static void main(String[] args) {
        //找数组中的最大元素
        int[] array = {3,5,1,40,55,20,80};
        int max = printMax(array);
        System.out.println(max);
    }

    public static int printMax(int[] array){
        int max = array[0];
        for(int i = 0; i < array.length - 1; i++){
            if(max > array[i+1]){
                max = array[i];
            }else {
                max = array[i + 1];
            }
        }
        return max;
    }

类似于 “打擂台” 这样的过程. 其中 max 变量作为 擂台, 比擂台上的元素大, 就替换上去, 否则就下一个对手

4.5、求数组中元素的平均值

public static void main(String[] args) {
        //求数组中元素的平均值
        int[] array = {3,5,1,40,55,20,80};
        double scr = avgSum(array);
        System.out.println(scr);
    }

    public static double avgSum(int[] array){
        double sum = 0.0;
        for(int i = 0; i < array.length; i++){
            sum = sum + array[i];
        }
        return sum / array.length;
    }

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

给定一个数组, 再给定一个元素, 找出该元素在数组中的位置

public static void main(String[] args) {
        //查找数组中指定元素(顺序查找)
        //返回下标,没有返回-1
        int[] array = {3,5,1,40,55,20,80};
        int k = 1;
        int ret = func(k,array);
        System.out.println("下标为:" + ret);
    }

    public  static int func(int k , int[] array){
        for(int i = 0; i < array.length; i++){
            if(k == array[i]){
                return i;
            }
        }
        return -1;
    }

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

针对有序数组, 可以使用更高效的二分查找.
啥叫有序数组?
有序分为 “升序” 和 “降序”
如 1 2 3 4 , 依次递增即为升序.
如 4 3 2 1 , 依次递减即为降序.
以升序数组为例, 二分查找的思路是先取中间位置的元素, 看要找的值比中间元素大还是小. 如果小, 就去左边找; 否则就去右边找.

public static void main(String[] args) {
        //二分查找
        int[] array = {1,2,3,4,5,6,7,8,9,10};
        //k 为要查找的数,找到返下标,找不到返回-1
        int k = 11;
        int ret = func(k,array);
        System.out.println(ret);
    }

    public static int func(int k, int[] array){
        int left = 0;
        int right = array.length - 1;
        while(left <= right){
            int mid = (left + right) / 2;
            if(array[mid] > k){
                //中间值 > k,说明 k 是在中间值的左边
                //此时的右值应该是中间值
                right = mid - 1;
            }else if(array[mid] < k){
                //中间值 < k,说明 k 是在中间值的右边
                //此时的左值应该是中间值
                left = mid + 1;
            }else{
                return mid;
            }
        }
        return -1;
    }

4.8、检查数组的有序性

给定一个整型数组, 判断是否该数组是有序的(升序)

public static void main(String[] args) {
        //给定一个整型数组, 判断是否该数组是有序的(升序)
        int[] array = {1,2,3,4,10,6,7,8,9,10};
        boolean b = funcSce(array);
        System.out.println(b);
    }

    public static boolean funcSce(int[] array){
        for(int i = 0; i < array.length - 1; i++){
            if(array[i] > array[i+1]){
                return false;
            }
        }
        return true;
    }

4.9、数组排序(冒泡排序)

public static void main(String[] args) {
        //数组排序(冒泡排序)
        int[] array = {10,9,8,7,4,5,6,2,3,1};
        int[] ret = funcMao(array);
        System.out.println(Arrays.toString(ret));
    }

    public static int[] funcMao(int[] array){
        //i 表示比较多少对
        for(int i = 0; i < array.length-1; i++){
            //j 表示比较的趟数
            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;
                }
            }
        }
        return array;
    }

4.10、数组逆序

给定一个数组, 将里面的元素逆序排列.
思路
设定两个下标, 分别指向第一个元素和最后一个元素. 交换两个位置的元素.
然后让前一个下标自增, 后一个下标自减, 循环继续即可.

public static void main(String[] args) {
        //数组逆序
        int[] array = {1,2,3,4,10,6,7,8,9,10};
        int[] ret = funcNixu(array);
        System.out.println(Arrays.toString(ret));
    }

    public static int[] funcNixu(int[] array){
        int left = 0;
        int right = array.length - 1;
        while(left <= right){
            int tmp = array[left];
            array[left] = array[right];
            array[right] = tmp;
            left++;
            right--;
        }
        return array;
    }

4.11、数组数字排列

给定一个整型数组, 将所有的偶数放在前半部分, 将所有的奇数放在数组后半部分
例如 {1, 2, 3, 4}
调整后得到 {4, 2, 3,1}

基本思路
设定两个下标分别指向第一个元素和最后一个元素.
用前一个下标从左往右找到第一个奇数, 用后一个下标从右往左找到第一个偶数, 然后交换两个位置的元素.依次循环即可.

public static void main(String[] args) {
        //给定一个整型数组, 将所有的偶数放在前半部分, 将所有的奇数放在数组后半部分
        int[] array = {1,2,3,4,10,6,7,8,9,10};
        int[] ret = swapNum(array);
        System.out.println(Arrays.toString(ret));
    }

    public static int[] swapNum(int[] array){
        int left = 0;
        int right = array.length - 1;
        while(left <= right){
            if(array[left] % 2 == 0){
                left++;
            }else if(array[right] % 2 != 0){
                right--;
            }else {
                int tmp = array[left];
                array[left] = array[right];
                array[right] = tmp;
            }
        }
        return array;
    }

五、 二维数组

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

5.1、基本语法

二维数组的创建

基本语法1

	数据类型[][] 数组名 = { 初始化数据 }

基本语法2

	 数据类型[][] 数组名 = new 数据类型[][]{ 初始化数据 }

基本语法3

	数据类型[][] 数组名 = new 数据类型[行数][列数]
public static void main(String[] args) {
        //和一位数组一样的定义方式
        //直接初始化数据内容
        int[][] array1 = {{1,2,3},{4,5,6}};
        //动态初始化
        int[][] array2 = new int[][]{{1,2,3},{4,5,6}};
        //静态初始化,没有初始化内容,默认为0
        int[][] array3 = new int[2][3];
    }

5.2、二维数组的打印

for循环打印

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();
        }
    }

在这里插入图片描述

for - each 打印数组

public static void main(String[] args) {
        int[][] array1 = {{1,2,3},{4,5,6}};
        for (int[] str : array1) {
            for (int n: str) {
                System.out.print(n + " ");
            }
            System.out.println();
        }
    }

在这里插入图片描述

使用java自带的函数来打印二维数组(deepToString)

在前面,我们使用了 Arrays.toString,将数组转换的字符串输出

二维数组也有对应的 方法: Arrays.deepToString(数组名)

public static void main(String[] args) {
        int[][] array1 = {{1,2,3},{4,5,6}};

        System.out.println(Arrays.deepToString(array1));
    }

在这里插入图片描述

5.3、一种特别的二维数组(不规则的二维数组)

第一种情况:

public static void main(String[] args) {
        int[][] array = {{1,2,},{4,5,6}};
        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();
        }
    }

在这里插入图片描述

第二种情况:不规则二维数组
二维数组可以省略列,不能省略行

public static void main(String[] args) {
        int[][] array = new int[2][];
        //手动给 二维数组 赋值
        //第一行有3个元素
        array[0] = new int[3];
        //第二行有4个元素
        array[1] = new int[4];
        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();
        }
    }

在这里插入图片描述

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

Später321

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

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

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

打赏作者

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

抵扣说明:

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

余额充值