> 文章列表 > 03-java数组的使用

03-java数组的使用

03-java数组的使用

概念

数组就是存储数据长度固定的容器,存储多个数据的数据类型要一致。

数组的定义格式

// 第一种格式
// 数据类型[] 数组名
int[] arr;        
double[] arr;      
char[] arr;// 第二种格式
// 数据类型 数组名[]
int arr[];
double arr[];
char arr[];

数组的动态初始化

什么是动态初始化

数组动态初始化就是只给定数组的长度,由系统给出默认初始化值

动态初始化格式

数据类型[] 数组名 = new 数据类型[数组长度];
int[] arr = new int[3];

动态初始化格式详解

  • 等号左边:

    • int:数组的数据类型

    • []:代表这是一个数组

    • arr:代表数组的名称

  • 等号右边:

    • new:为数组开辟内存空间
    • int:数组的数据类型
    • []:代表这是一个数组
    • 5:代表数组的长度

代码 :

package com.itheima.array;public class Demo2Array {/*数组的动态初始化:在初始化的时候, 需要手动指定数组的长度, 系统会为数组容器分配初始值.动态初始化格式:数据类型[] 数组名 = new 数据类型[数组的长度];注意:打印数组变量的时候, 会打印出数组的内存地址[I@10f87f48 :@ : 分隔符[ : 当前的空间是一个数组类型I : 当前数组容器中所存储的数据类型10f87f48 : 十六进制内存地址0 1 2 3 4 5 6 7 8 9 a b c d e f*/public static void main(String[] args) {// 数据类型[] 数组名 = new 数据类型[数组的长度];// 通过new关键字创建了一个int类型的数组容器, 该容器可以存储5个int类型的整数, 该容器被arr数组变量所记录int[] arr = new int[5];// [I@10f87f48System.out.println(arr);byte[] bArr = new byte[3];// [B@b4c966aSystem.out.println(bArr);}
}

数组的静态初始化

什么是静态初始化

在创建数组时,直接将元素确定

静态初始化格式

完整版格式  
数据类型[] 数组名 = new 数据类型[]{元素1,元素2,...};简化版格式  
数据类型[] 数组名 = {元素1,元素2,...};

数组元素访问

什么是索引

每一个存储到数组的元素,都会自动的拥有一个编号,从0开始。

这个自动编号称为数组索引(index),可以通过数组的索引访问到数组中的元素。

访问数组元素格式

数组名[索引];

内存概述

内存是计算机中的重要原件,临时存储区域,作用是运行程序。

我们编写的程序是存放在硬盘中的,在硬盘中的程序是不会运行的。

必须放进内存中才能运行,运行完毕后会清空内存。

Java虚拟机要运行程序,必须要对内存进行空间的分配和管理。

数组操作的两个常见问题

索引越界异常

  • 出现原因
    数组长度为3,索引范围是0~2,但是我们却访问了一个3的索引。
    程序运行后,将会抛出ArrayIndexOutOfBoundsException 数组越界异常。在开发中,数组的越界异常是不能出现的,一旦出现了,就必须要修改我们编写的代码。
public class ArrayDemo {public static void main(String[] args) {int[] arr = new int[3];System.out.println(arr[3]);}
}
  • 解决方案
    将错误的索引修改为正确的索引范围即可!

空指针异常

  • 出现原因
    arr = null 这行代码,意味着变量arr将不会在保存数组的内存地址,也就不允许再操作数组了,因此运行的时候会抛出 NullPointerException 空指针异常。在开发中,空指针异常是不能出现的,一旦出现了,就必须要修改我们编写的代码。
public class ArrayDemo {public static void main(String[] args) {int[] arr = new int[3];//把null赋值给数组arr = null;System.out.println(arr[0]);}
}
  • 解决方案
    给数组一个真正的堆内存空间引用即可!

数组常用操作

数组遍历

  • 数组遍历:就是将数组中的每个元素分别获取出来,就是遍历。遍历也是数组操作中的基石。
  • 其实遍历的就两种,对象和数组.只是哪个遍历适用于哪个而已,后面会总结一下
package com.demo;import java.util.ArrayList;
import java.util.Arrays;
import java.util.Iterator;public class demo {public static void main(String[] args) {// 数组遍历方式String[] arrs = {"a", "b", "c"};// forfor (int i = 0; i < arrs.length; i++) {System.out.println(arrs[i]);}System.out.println("================");// for增强for (String str : arrs) {System.out.println(str);}System.out.println("================");// Iterator(迭代器)Iterator<String> iterator = Arrays.stream(arrs).iterator();while (iterator.hasNext()){String next = iterator.next();System.out.println(next);}System.out.println("================");//streamArrays.stream(arrs).forEach(System.out::println);// ArraysSystem.out.println(Arrays.toString(arrs));}
}

数组过滤/查找

常见过滤

代码实现:

public static void main(String[] args) {String[] arrs = {"a", "b", "c"};for (String arr : arrs) {if(!"a".equals(arr)){System.out.println("打印出不是a的值:"+ arr);}}
}

二分查找 (理解)

注意事项:有一个前提条件,数组内的元素一定要按照大小顺序排列,如果没有大小顺序,是不能使用二分查找法的

二分查找概述

  • 查找指定元素在数组中的位置时,以前的方式是通过遍历,逐个获取每个元素,看是否是要查找的元素,这种方式当数组元素较多时,查找的效率很低
  • 二分查找也叫折半查找,每次可以去掉一半的查找范围,从而提高查找的效率

需求

  • 在数组{1,2,3,4,5,6,7,8,9,10}中,查找某个元素的位置

实现步骤

  1. 定义两个变量,表示要查找的范围。默认min = 0 ,max = 最大索引
  2. 循环查找,但是min <= max
  3. 计算出mid的值
  4. 判断mid位置的元素是否为要查找的元素,如果是直接返回对应索引
  5. 如果要查找的值在mid的左半边,那么min值不变,max = mid -1.继续下次循环查找
  6. 如果要查找的值在mid的右半边,那么max值不变,min = mid + 1.继续下次循环查找
  7. 当min > max 时,表示要查找的元素在数组中不存在,返回-1.

代码实现

public class MyBinarySearchDemo {public static void main(String[] args) {int [] arr = {1,2,3,4,5,6,7,8,9,10};int number = 11;//1,我现在要干嘛? --- 二分查找//2.我干这件事情需要什么? --- 数组 元素//3,我干完了,要不要把结果返回调用者 --- 把索引返回给调用者int index = binarySearchForIndex(arr,number);System.out.println(index);}private static int binarySearchForIndex(int[] arr, int number) {//1,定义查找的范围int min = 0;int max = arr.length - 1;//2.循环查找 min <= maxwhile(min <= max){//3.计算出中间位置 midint mid = (min + max) >> 1;//mid指向的元素 > numberif(arr[mid] > number){//表示要查找的元素在左边.max = mid -1;}else if(arr[mid] < number){//mid指向的元素 < number//表示要查找的元素在右边.min = mid + 1;}else{//mid指向的元素 == numberreturn mid;}}//如果min大于了max就表示元素不存在,返回-1.return -1;}}

数据排序

冒泡排序

步骤

比较相邻的元素。如果第一个比第二个大,就交换他们两个。

动图演示

03-java数组的使用

代码实现

public class Demo01 {public static void main(String[] args) {int[] arr = {3,4,1,2};int[] ints = bubbleSort(arr);Arrays.stream(ints).forEach(System.out::println);System.out.println(Arrays.toString(bubbleSort(arr)));}// 返回一个对原数组没有影响的排好序的新数组public static int[] bubbleSort(int[] sourceArray)   {// 对 arr 进行拷贝,不改变参数内容int[] arr = Arrays.copyOf(sourceArray, sourceArray.length);for (int i = 1; i < arr.length; i++) {for (int j = 0; j < arr.length - i; j++) {if (arr[j] > arr[j + 1]) {int tmp = arr[j];arr[j] = arr[j + 1];arr[j + 1] = tmp;}}}return arr;}
}

选择排序

步骤

  1. 首先在未排序序列中找到最小(大)元素,存放到排序序列的起始位置。
  2. 再从剩余未排序元素中继续寻找最小(大)元素,然后放到已排序序列的末尾。

动图演示

03-java数组的使用

代码实现

public class Demo01 {public static void main(String[] args) {int[] arr = {3,4,1,2};int[] ints = selectSort(arr);Arrays.stream(ints).forEach(System.out::println);}public static int[] selectSort(int[] arrs){int minIndex,temp;for (int i = 0; i < arrs.length-1; i++) {  // 注意,这里需要减一minIndex = i; // 默认最小值元素的索引for (int j = i+1; j < arrs.length; j++) {if(arrs[minIndex] > arrs[j]){  // 依次查找最小值的索引minIndex = j;}}if(minIndex != i){   // 如果最小值的索引不一样,交换位置temp = arrs[minIndex];arrs[minIndex] = arrs[i];arrs[i] = temp;}}return arrs;}
}

插入排序

步骤

将第一待排序序列第一个元素看做一个有序序列,把第二个元素到最后一个元素当成是未排序序列。

从头到尾依次扫描未排序序列,将扫描到的每个元素插入有序序列的适当位置。(如果待插入的元素与有序序列中的某个元素相等,则将待插入元素插入到相等元素的后面。

说真的,说的有点迷糊,按照我的理解来说,举个例子, 玩牌,我们玩牌的时候怎么对牌进行排序

  1. 从所有牌中拿一个牌
  2. 第二次拿牌,比较和第一章那个大,然后排序
  3. 第三次,和前面那的两张牌比较,插入指定位置。。。
  4. 最后一次,拿到最后一张牌,和你手里的牌对比,插入指定位置

动图演示

03-java数组的使用

代码实现

public class Demo01 {public static void main(String[] args) {int[] arr = {3,4,1,2};int[] ints = insertSort(arr);Arrays.stream(ints).forEach(System.out::println);}public static int[] insertSort(int[] sourceArray) {// 对 arr 进行拷贝,不改变参数内容int[] arr = Arrays.copyOf(sourceArray, sourceArray.length);// 从下标为1的元素开始选择合适的位置插入,因为下标为0的只有一个元素,默认是有序的for (int i = 1; i < arr.length; i++) {// 记录要插入的数据int tmp = arr[i];// 从已经排序的序列最右边的开始比较,找到比其小的数int j = i;while (j > 0 && tmp < arr[j - 1]) {arr[j] = arr[j - 1];j--;}// 存在比其小的数,插入if (j != i) {arr[j] = tmp;}}return arr;}
}

数组元素求和/字符串拼接

    public static void main(String[] args) {// 求和int[] arr = {1,2,3,4};int total = 0;for (int i : arr) {total += i;}System.out.println("合为:"+total);String[] strs = {"a","b","c","d"};String str = "";for (String s : strs) {str += s;}System.out.println("拼接后的字符串为:"+str);}

数组获取最大值

  • 最大值获取:从数组的所有元素中找出最大值。

  • 实现思路:

    • 定义变量,保存数组0索引上的元素
    • 遍历数组,获取出数组中的每个元素
    • 将遍历到的元素和保存数组0索引上值的变量进行比较
    • 如果数组元素的值大于了变量的值,变量记录住新的值
    • 数组循环遍历结束,变量保存的就是数组中的最大值
  • 代码实现:

package com.itheima.test;import java.util.Scanner;public class Test2Array {/*需求: 从数组中查找最大值int[] arr = {12,45,98,73,60};实现步骤:1. 假设数组中的第一个元素为最大值2. 遍历数组, 获取每一个元素, 准备进行比较3. 如果比较的过程中, 出现了比max更大的, 让max记录更大的值4. 循环结束后, 打印最大值.*/public static void main(String[] args) {int[] arr = {12,45,98,73,60};// 1. 假设数组中的第一个元素为最大值int max = arr[0];// 2. 遍历数组, 获取每一个元素, 准备进行比较for(int i = 1; i < arr.length; i++){// 3. 如果比较的过程中, 出现了比max更大的, 让max记录更大的值if(arr[i] > max){max = arr[i];}}//  4. 循环结束后, 打印最大值.System.out.println("max:" + max);}
}

Arrays (应用)

Arrays的常用方法

方法名 说明
public static String toString(int[] a) 返回指定数组的内容的字符串表示形式
public static void sort(int[] a) 按照数字顺序排列指定的数组
public static int binarySearch(int[] a, int key) 利用二分查找返回指定元素的索引

示例代码

public class MyArraysDemo {public static void main(String[] args) {//        public static String toString(int[] a)    返回指定数组的内容的字符串表示形式//        int [] arr = {3,2,4,6,7};//        System.out.println(Arrays.toString(arr));//        public static void sort(int[] a)      按照数字顺序排列指定的数组//        int [] arr = {3,2,4,6,7};//        Arrays.sort(arr);//        System.out.println(Arrays.toString(arr));//        public static int binarySearch(int[] a, int key) 利用二分查找返回指定元素的索引int [] arr = {1,2,3,4,5,6,7,8,9,10};int index = Arrays.binarySearch(arr, 0);System.out.println(index);//1,数组必须有序//2.如果要查找的元素存在,那么返回的是这个元素实际的索引//3.如果要查找的元素不存在,那么返回的是 (-插入点-1)//插入点:如果这个元素在数组中,他应该在哪个索引上.}}
  • 工具类设计思想
    1. 构造方法用 private 修饰
    2. 成员用 public static 修饰

二维数组概述

概述 : 二维数组也是一种容器,不同于一维数组,该容器存储的都是一维数组容器

二维数组动态初始化

动态初始化格式:数据类型[][] 变量名 = new 数据类型[m][n];
m表示这个二维数组,可以存放多少个一维数组
n表示每一个一维数组,可以存放多少个元素
package com.itheima.demo;public class Demo1Array {/*动态初始化格式:数据类型[][] 变量名 = new 数据类型[m][n];m表示这个二维数组,可以存放多少个一维数组n表示每一个一维数组,可以存放多少个元素*/public static void main(String[] args) {// 数据类型[][] 变量名 = new 数据类型[m][n];int[][] arr = new int[3][3];/*[[I@10f87f48@ : 分隔符10f87f48 : 十六进制内存地址I : 数组中存储的数据类型[[ : 几个中括号就代表的是几维数组*/System.out.println(arr);/*二维数组存储一维数组的时候, 存储的是一维数组的内存地址*/System.out.println(arr[0]);System.out.println(arr[1]);System.out.println(arr[2]);System.out.println(arr[0][0]);System.out.println(arr[1][1]);System.out.println(arr[2][2]);// 向二维数组中存储元素arr[0][0] = 11;arr[0][1] = 22;arr[0][2] = 33;arr[1][0] = 11;arr[1][1] = 22;arr[1][2] = 33;arr[2][0] = 11;arr[2][1] = 22;arr[2][2] = 33;// 从二维数组中取出元素并打印System.out.println(arr[0][0]);System.out.println(arr[0][1]);System.out.println(arr[0][2]);System.out.println(arr[1][0]);System.out.println(arr[1][1]);System.out.println(arr[1][2]);System.out.println(arr[2][0]);System.out.println(arr[2][1]);System.out.println(arr[2][2]);}
}

二维数组静态初始化

代码实现 :

package com.itheima.demo;public class Demo3Array {/*完整格式:数据类型[][] 变量名 = new 数据类型[][]{ {元素1, 元素2...} , {元素1, 元素2...} ...};简化格式: 数据类型[][] 变量名 = { {元素1, 元素2...} , {元素1, 元素2...} ...};*/public static void main(String[] args) {int[] arr1 = {11,22,33};int[] arr2 = {44,55,66};int[][] arr = {{11,22,33}, {44,55,66}};System.out.println(arr[0][2]);int[][] array = {arr1,arr2};System.out.println(array[0][2]);}
}

二维数组遍历

需求 :

已知一个二维数组 arr = {{11, 22, 33}, {33, 44, 55}};

遍历该数组,取出所有元素并打印

步骤 :

  1. 遍历二维数组,取出里面每一个一维数组

  2. 在遍历的过程中,对每一个一维数组继续完成遍历,获取内部存储的每一个元素

代码实现 :

package com.itheima.test;public class Test1 {/*需求:已知一个二维数组 arr = {{11, 22, 33}, {33, 44, 55}};遍历该数组,取出所有元素并打印步骤:1. 遍历二维数组,取出里面每一个一维数组2. 在遍历的过程中,对每一个一维数组继续完成遍历,获取内部存储的每一个元素*/public static void main(String[] args) {int[][] arr = {{11, 22, 33}, {33, 44, 55}};// 1. 遍历二维数组,取出里面每一个一维数组for (int i = 0; i < arr.length; i++) {//System.out.println(arr[i]);// 2. 在遍历的过程中,对每一个一维数组继续完成遍历,获取内部存储的每一个元素//int[] temp = arr[i];for (int j = 0; j < arr[i].length; j++) {System.out.println(arr[i][j]);}}}
}

二维数组求和

需求 :

某公司季度和月份统计的数据如下:单位(万元)
第一季度:22,66,44
第二季度:77,33,88
第三季度:25,45,65
第四季度:11,66,99

步骤 :

  1. 定义求和变量,准备记录最终累加结果
  2. 使用二维数组来存储数据,每个季度是一个一维数组,再将4个一维数组装起来
  3. 遍历二维数组,获取所有元素,累加求和
  4. 输出最终结果

代码实现 :

package com.itheima.test;public class Test2 {/*需求:某公司季度和月份统计的数据如下:单位(万元)第一季度:22,66,44第二季度:77,33,88第三季度:25,45,65第四季度:11,66,99步骤:1. 定义求和变量,准备记录最终累加结果2. 使用二维数组来存储数据,每个季度是一个一维数组,再将4个一维数组装起来3. 遍历二维数组,获取所有元素,累加求和4. 输出最终结果*/public static void main(String[] args) {// 1. 定义求和变量,准备记录最终累加结果int sum = 0;// 2. 使用二维数组来存储数据,每个季度是一个一维数组,再将4个一维数组装起来int[][] arr = { {22,66,44} , {77,33,88} , {25,45,65} , {11,66,99}};// 3. 遍历二维数组,获取所有元素,累加求和for (int i = 0; i < arr.length; i++) {for(int j = 0; j < arr[i].length; j++){sum += arr[i][j];}}// 4. 输出最终结果System.out.println(sum);}
}