JavaSE学习总结(四)二维数组概述/二维数组动态初始化/二维数组静态初始化/二维数组遍历/递归解决问题的思想

一、二维数组

(一)二维数组概述

二维数组其实就是每一个元素为一维数组的数组。

(二)二维数组初始化格式

1.动态初始化

1.1 二维数组格式1

数据类型[][] 变量名 = new 数据类型[m][n];
m表示这个二维数组有多少个一维数组 必须写上
n表示每一个一维数组的元素个数 可选

举例:
int[][] arr = new int[3][2];
定义了一个二维数组arr
这个二维数组有3个一维数组,名称是arr[0],arr[1],arr[2]
每个一维数组有2个元素,可以通过arr[m][n]来获取
表示获取第m+1个一维数组的第n+1个元素

注意事项
(1)以下格式也可以表示二维数组

  • 数据类型 数组名[][] = new 数据类型[m][n];
  • 数据类型[] 数组名[] = new 数据类型[m][n];
    这两种格式不推荐使用

(2)注意下面定义的区别

int x,y;
int[] x,y[];

区别是:
int[] x,y[];
定义了两个数组 一个是一维数组x 一个是二维数组y

案例演示

public class ArrayDemo {
    public static void main(String[] args) {
        //二维数组:数组中的元素是一维数组,数组嵌套数组
        //动态初始化

        //3 表示这个二维数组里面,放了3个一维数组
        //2 表示二维数组中的一维数组的长度
        int[][] arr=new int[3][2];
        arr[0]=new int[]{10,20};//为第一个一维数组赋值
        System.out.println(arr[0]);//二维数组里第一个一维数组的地址
        System.out.println(arr[1]);//二维数组里第二个一维数组的地址
        System.out.println(arr[0][0]);//输出第一个一维数组的第一个元素值
        System.out.println(arr[0][1]);//输出第一个一维数组的第二个元素值
        System.out.println(arr.length); //二维数组的长度
        System.out.println(arr[0].length);//二维数组中的第一个一维数组的长度
    }
}

JavaSE学习总结(四)二维数组概述/二维数组动态初始化/二维数组静态初始化/二维数组遍历/递归解决问题的思想_第1张图片
内存解析
JavaSE学习总结(四)二维数组概述/二维数组动态初始化/二维数组静态初始化/二维数组遍历/递归解决问题的思想_第2张图片
  首先,程序编译好后生成了字节码文件(.class文件),JVM将字节码文件加载进内存的方法区,而main方法是程序的入口,需要被执行,于是调用main方法进栈执行。接着,执行第一句代码,创建了一个长度为3的int型二维数组,其每个元素为长度为2的一维数组。于是堆内存为该二维数组开辟空间,并将3个元素都初始化为null,内存空间地址为0x12345678(此处随意写的一个地址),然后再初始化3个长度为2的一维数组,开辟完空间以后将每个一维数组的引用覆盖掉二维数组的初始值null,于是这个二维数组的每个元素都指向对应的一维数组。再将0x12345678赋给二维数组的引用arr,则名为arr的这个数组便指向了地址为0x12345678的空间。main方法执行完毕后,main方法弹栈,此时就没有引用指向堆内存中地址为0x12345678的空间了,于是最后垃圾回收器回收了该空间,释放内存。

1.2 二维数组格式2

数据类型[][] 变量名 = new 数据类型[m][];
m表示这个二维数组有多少个一维数组
这种格式没有直接给出一维数组的元素个数,可以动态的给出。

举例:

int[][] arr = new int[3][];
arr[0] = new int[2];//二维数组里的第一个一维数组长度为2
arr[1] = new int[3];//二维数组里的第二个一维数组长度为3
arr[2] = new int[1];//二维数组里的第三个一维数组长度为1

案例演示

public class ArrayDemo2 {
    public static void main(String[] args) {
        int[][] arr=new int[3][];
        System.out.println(arr);//二维数组地址
        System.out.println(arr[0]);//二维数组中第一个元素的值(初值null)
        System.out.println(arr[1]);
        System.out.println(arr[2]);
        arr[0]=new int[3];
        arr[1]=new int[5];
        arr[2]=new int[4];
        System.out.println(arr[0]);//动态赋值后第一个一维数组地址
        System.out.println(arr[1]);
        System.out.println(arr[2]);
    }
}

JavaSE学习总结(四)二维数组概述/二维数组动态初始化/二维数组静态初始化/二维数组遍历/递归解决问题的思想_第3张图片
内存解析:
JavaSE学习总结(四)二维数组概述/二维数组动态初始化/二维数组静态初始化/二维数组遍历/递归解决问题的思想_第4张图片
这种格式,没有直接给出一维数组的元素个数,因此在创建二维数组时只会为二维数组开辟空间并初始化,不会为一维数组开辟空间和初始化。

注意事项:数组的长度不宜过长,否则会报堆内存不足的错误

public class Demo1 {
    public static void main(String[] args) {
        int[][] arr=new int[999999999][];
    }
}

在这里插入图片描述

2.静态初始化

2.1 二维数组格式3

数据类型[][] 变量名 = new 数据类型[][]{{元素…},{元素…},{元素…}...};

简化版:
数据类型[][] 变量名 = {{元素…},{元素…},{元素…}};
这个格式属于静态初始化:由我们指定具体的元素值,由系统给分配长度

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

案例演示

public class ArrayDemo {
    public static void main(String[] args) {
        //二维数组静态初始化
        int[][] arr=new int[][]{{2,4},{10,30},{10,30,40},{10,1}};
        System.out.println(arr.length);
        System.out.println(arr[3][1]);
        //简写方式
        int[][] arr2 ={{2, 4}, {10, 30}, {10, 30, 40}, {10, 1},{2,5}};
        System.out.println(arr2.length);
        System.out.println(arr2[2][2]);
    }
}

在这里插入图片描述

(三)二维数组的遍历

案例演示
1.

public class ArrayDemo {
    public static void main(String[] args) {
        int[][] arr = {{2, 4}, {10, 30}, {10, 30, 40}, {10, 1}};
        //二维数组的遍历
        //外循环控制的是二维数组的长度,其实就是一维数组的个数。
		//内循环控制的是一维数组的长度。
        for (int i = 0; i < arr.length; i++) {
            for (int j = 0; j < arr[i].length; j++) {
                System.out.println(arr[i][j]);
            }
        }
    }
}

JavaSE学习总结(四)二维数组概述/二维数组动态初始化/二维数组静态初始化/二维数组遍历/递归解决问题的思想_第5张图片
2.公司年销售额求和
某公司按照季度和月份统计的数据如下:单位(万元)
第一季度:22,66,44
第二季度:77,33,88
第三季度:25,45,65
第四季度:11,66,99

public class ArrayDemo {
    public static void main(String[] args) {
        int[][] arr={{22,66,44},{77,33,88},{25,45,65},{11,66,99}};
        int sum=0;
        for (int i = 0; i < arr.length; i++) {
            for (int j = 0; j < arr[i].length; j++) {
                sum+=arr[i][j];
            }
        }
        System.out.println("公司年销售额为:"+sum);
    }
}

在这里插入图片描述

3.需求:打印杨辉三角形(行数可以键盘录入)
1
1 1
1 2 1
1 3 3 1
1 4 6 4 1
1 5 10 10 5 1

import java.util.Scanner;

public class Demo {
    public static void main(String[] args) {
        Scanner sc = new Scanner(System.in);
        System.out.println("请输入一个数字");
        int x = sc.nextInt();
        int[][] a=new int[x][x];
        for(int i=0;i<a.length;i++){//任何一行的第一列和最后一列都是1
            a[i][0]=1;
            a[i][i]=1;
        }
        //从第三行开始,每一个数据是它上一行的前一列和它上一行的本列之和。
        for(int i=2;i<a.length;i++){
            for(int j=1;j<i;j++){//第一列和最后一列已赋值
                a[i][j]=a[i-1][j-1]+a[i-1][j];
            }
        }
        //遍历二维数组 输出结果  
        for (int i = 0; i < a.length; i++) {
            for(int j=0;j<=i;j++){
                System.out.print(a[i][j]+" ");
            }
            System.out.println();
        }
    }
}

JavaSE学习总结(四)二维数组概述/二维数组动态初始化/二维数组静态初始化/二维数组遍历/递归解决问题的思想_第6张图片

(四)思考题

看程序写结果,并画内存图解释

public static void main(String[] args) {
		int a = 10;
		int b = 20;
		System.out.println("a: " + a + ",b: " + b);
		change(a,b);
		System.out.println("a: " + a + ",b: " + b);
		int[] arr = {1,2,3,4,5};
		change(arr);
		System.out.println(arr[1]);
}
public static void change(int a,int b)  {
		System.out.println("a: " + a + ",b: " + b);
		a = b;
		b = a + b;
		System.out.println("a: " + a + ",b: " + b);
}
public static void change(int[] arr){
		for(int x = 0 ; x < arr.length ; x++){
			if(arr[x]%2 == 0){
				arr[x] *= 2;
			}
		}
}

答案
JavaSE学习总结(四)二维数组概述/二维数组动态初始化/二维数组静态初始化/二维数组遍历/递归解决问题的思想_第7张图片
内存解析
JavaSE学习总结(四)二维数组概述/二维数组动态初始化/二维数组静态初始化/二维数组遍历/递归解决问题的思想_第8张图片
首先main方法被调用进栈,然后定义了两个变量a和b并赋了值。当main方法调用change(int a,int b)方法时,此方法进栈,并执行其中代码,修改了两个变量的值,执行完以后便出栈了。返回主方法中继续执行代码,此时输出a和b的值时只能找到主方法中的a和b,因此a和b的值仍然不变。接着创建一个int型数组并赋值,在堆内存中开辟了空间并赋上各元素值,并将该空间的地址赋给数组的引用arr,于是arr便指向该空间的数组。当调用change(int[] arr)时,此方法进栈,并执行其中代码,修改了数组中两个元素的值,执行完以后方法弹栈,但堆内存中数据的改动仍然保留,返回主方法继续执行代码,找到arr指向地址的数组,找到索引为1的元素,输出。
基本数据类型,作为参数传递,形参的改变,不影响实参
引用数据类型,作为参数传递,形参的改变,会影响实参

二、递归

(一)递归概述

方法定义中调用该方法本身的现象

递归注意事项:

  • 要有出口,否则就是死递归,会造成栈内存溢出
  • 递归次数不能太多,否则也会造成栈内存溢出

死递归:

public class Demo1 {
    public static void main(String[] args) {
        test();
    }
    public static void test(){
        System.out.println("这是一个死递归");
        test();
    }
}

在这里插入图片描述

递归在生活中的举例:
从前有座山,山里有座庙,庙里有个老和尚,老和尚给小和尚在讲故事:从前有座山,山里有座庙…

(二)递归解决问题的思想

递归解决问题的思想即“拆分合并
也就是将一个大问题拆分成一个个小问题,解决完小问题后再合并,便解决了大问题。

案例演示

1.求5的阶乘

方法1:利用循环

public class MyTest {
    public static void main(String[] args) {
        //问题求 5的阶乘  5!=5*4*3*2*1;
        //循环做
        int r=1;
        for (int i = 1; i <= 5; i++) {
            r*=i;
        }
        System.out.println("结果是"+r);
    }
}

在这里插入图片描述
方法2:利用递归
JavaSE学习总结(四)二维数组概述/二维数组动态初始化/二维数组静态初始化/二维数组遍历/递归解决问题的思想_第9张图片

public class MyTest {
    public static void main(String[] args) {
        //求 5的阶乘
        //用递归来做
       int r= jieCheng(5);
        System.out.println("结果是"+r);

    }
    public static int jieCheng(int i) {
        if(i==1){
            return 1;
        }else{
            return i*jieCheng(i-1);
        }
    }
}

在这里插入图片描述
JavaSE学习总结(四)二维数组概述/二维数组动态初始化/二维数组静态初始化/二维数组遍历/递归解决问题的思想_第10张图片
递归和循环的区别和联系

递归算法:
优点:代码简洁、清晰,并且容易验证正确性。
缺点:它的运行需要较多次数的方法调用,如果调用层数比较深,会对执行效率有一定影响。并且调用次数过多会出现堆内存溢出的现象。

循环算法:
优点:速度快,结构简单。
缺点:并不能解决所有的问题。

用循环能实现的,递归一般可以实现,但是能用递归实现的,循环不一定能。

2.兔子问题(斐波那契数列)
题目:有一对兔子,从出生后第3个月起每个月都生一对兔子,小兔子长到第三个月后每个月又生一对兔子,假如兔子都不死,问第二十个月的兔子对数为多少?

分析:由此可见兔子对象每个月的对数分别是:1 , 1 , 2 , 3 , 5 , 8, 13 …
从中找到规律:前两个数都是1,从第三个数开始,这个数等于前两个数之和 (斐波那契数列)

方法1:

public class MyTest {
    public static void main(String[] args) {
		//采用数组方法来做,到第20个月有多少对兔子
        int[] arr=new int[20];
        arr[0]=1;
        arr[1]=1;
        for (int i =2; i < arr.length; i++) {
            arr[i]=arr[i-1]+arr[i-2];
        }
        System.out.println("兔子的对数"+arr[19]);
    }
}

在这里插入图片描述

方法2:递归

public class MyTest2 {
    public static void main(String[] args) {
        //递归来做
        int sum = sumRabbit(20);
        System.out.println("兔子的对数" + sum);
    }
    public static int sumRabbit(int i) {
        if (i == 1 || i == 2) {
            return 1;
        } else {
            return sumRabbit(i - 1) + sumRabbit(i - 2);
        }
    }
}

在这里插入图片描述

你可能感兴趣的:(#,JavaSE)