java为什么要学习数组_Java千问:七个问题帮助初学者深入理解Java数组

几乎所有的高级语言当中,都提供了一种叫做”数组”的东西,Java语言当然也不例外。我们通过数组可以很方便的存储和管理一组数据。因为在Java语言当中使用数组非常的方便,所以导致很多初学者忽略了对数组的深入学习,本文就通过七个问题,来帮助初学者深入理解一下Java语言的数组到底是怎么回事。

一、数组是变量的简单叠加吗?

我们在学习Java的时候,知道一个int类型的变量可以存储一个整数,而一个int类想的数组可以存储多个整数。于是很多人认为数组只不过是变量的简单叠加而已,无非是变量存1个数,数组存多个数。其实不然,Java语言是把数组当作一个“对象”来看待的。我们来看下面的代码

a3d8ec5e6c81dd4011f4ed17e297539a.png

我们可以看到,在代码中声明了int型变量a和int型数组b,b的长度为1。a和b都只能存储1个int型数据,似乎它们之间没什么区别。但是请注意:a和b之间其实是有本质区别的,a是一个基础数据类型的变量,而b则是一个对象!正因为它是一个对象,所以我们才可以调用到它的getClass()方法。那么紧接着小伙伴们会问:b既然是一个对象,那么这个对象是什么类型的呢?我们可以通过“b.getClass().getName()”获得它类型的名称是” [I”。读者也可以通过这种方式,获得其他几种基础数据类型数组的类型名称。最重要的是,我们必须搞清楚:数组不是变量的简单叠加!即便是基础数据类型的数组,也是以对象的形式存在的。

二、为什么数组的下标是从0开始的?

同很多编程语言一样,Java语言数组的下标也是从0开始的。很多初学者都不理解,为什么数组下标要从0而不是从1开始?这太不符合我们的日常计数习惯了。为了说清楚这个问题,我们还是先看一段代码

4a5dfeaa33d261bd122772543cf90585.png

在这段代码中,创建了一个长度为5的数组并命名为a。在程序实际运行的时候,会分配一组连续的空间,这组空间中可以存储5个int型数据。为了能够准确的从这组空间的任意一个单元中找到数据,虚拟机必须能够对这组空间中任意一个单元做出识别。那么如何识别每个单元呢?虚拟机可以把每个单元都进行单独的命名,这样做当然能够达到目的,但是这么做跟使用变量就没什么区别了,更体现不出这组数据是一个连续的整体。实际的情况是:虚拟机给这组空间的第一个单元命名为a,如果希望找到a本身,那么就直接用a[0]来表示,其中方括号中的0表示偏移量。偏移量为0,就表示找的就是a这个空间中的数据,而a[1]表示以a为基准点,偏移1个单元,这样就找到了数组中第2个元素,以此类推,想找到数组中第5个元素,就以a这个单元为基准点,再偏移4个单元即可。因此我们就可以知道,数组的下标本质上并不是数组元素的编号,而是以数组首元素为基准点所偏移的量。这就是数组下标从0开始的原因。

三、数组的长度为什么可以用变量表示?

我们还是先来看一段代码

cf035fba1c9cfbb1325df4bc0da41d42.png

这段代码中,在创建数组的时候,以变量表示了这个数组的长度,这种做法放到Java语言中似乎很常见,但是如果使用C语言编程,在创建数组的时候,数组的长度必须是一个固定的值,不能用变量来表示。很多人似乎对这个问题不以为然,只是简单的认为这只是Java和C语言之间一点简单的语法差别而已。

其实问题远不是这么简单。这个简单的语法差别其实体现出了Java与C语言之间编译和运行机制本质的不同。变量是运行时才被赋值的,Java语言允许把数组的长度以变量的形式来表示,其背后的根本原因就是允许数组一直等到”运行时”才把长度确定下来。也就是说,程序员可以在编码的时候,不用规定数组的长度,等到程序实际运行的时候,根据实际需要去确定数组的长度,这样大大增加了程序的灵活性。

举个例子:如果我们编写程序,要求用户输入几个数字保存到数组中,然后进行排序,用户输入多少个数字不确定。如果我们用Java语言去完成这个程序,只要先让用户输入他想对多少个数字进行排序,然后把数字的个数保存到一个变量中,接着以这个变量的值作为长度创建数组就可以。但是如果是用C语言来完成程序,那么程序员必须在编写代码的时候就把数组的长度定下来,而不能等到程序运行的时候再去确定。因为C语言不能动态的在运行时确定数组的长度,必须在编译阶段就把长度确定下来。但是程序员在编码的时候并不确定用户要对多少数字完成排序,于是就只能以经验推测,创建一个他认为”足够大”的数组来存放排序的数字。如果创建的数组太大,则浪费空间,如果数组太小,则无法保存全部数字。

说了这么多,重点其实就一句话:Java语言的数组可以在”运行时”确定长度!并且这个特点大大的增加了程序的灵活性。

四、Java数组的长度可以为0吗?

这个问题很简单,只要写一段代码就可以得到答案:可以!关键问题来了:长度为0的数组不能存数据,这种数组有什么意义呢?前面说过,Java语言允许在程序运行过程中动态确定数组长度。那么我们就可以设想这样的一个场景:要求编码查询出考试总分在700分以上的学员,并且把他们的姓名存储到一个字符串数组中。假如经过查询之后,发现没有成绩在700分以上的学员,该如何表示这个查询结果呢?如果查询方法返回null来表示没有查到符合条件的数据,会对后续程序带来一定的风险,因为毕竟空对象可能会导致后面的处理代码中抛出空指针异常。而用一个长度为0的数组来表示查询结果就安全多了,仅能体现出没有查到符合条件的数据,又降低了抛出异常的风险。

五、Java语言中的数组可以扩大容量吗?

答案是不可以!因为数组一旦分配了空间之后,如果想在原来空间的基础上扩大容量,就如同是扩张地盘,势必会影响到内存中其他数据的存储,所以Java语言不允许数组扩大容量。网上有很多资料,讲解了如何扩大数组的容量,并给出了实现代码。如果你仔细去看这些代码,你会发现,这些所谓扩大数组容量的算法,其实现过程都是新创建一个更大的数组,然后把原数组中的数据拷贝到新数组中,最终返回那个新创建的数组。因此这种所谓的扩容算法其实并不是真正的扩容。

六、可以创建抽象类数组吗?

还是看代码

37b04b26ee9f7083d6de5f662471f189.png

代码中创建了一个抽象类A,并且在main()方法中创建了一个A类型的数组。问题又来了:抽象类A不能创建对象,而现在不仅创建了对象,并且还是一组!这是怎么回事?其实这也是很多初学者在理解引用类型数据数组的一个误区:认为数组中存放的某种类型的对象。其实不然,引用数据类型的数组中,存放的并不是对象,而只是能指向这种对象的引用而已。就拿上面这段代码来说,数组a中并没有存放5个A类型的对象,而只是存放了5个能够指向A类型对象的引用,这些引用在数组初始化的过程中,都指向空对象(null),所以千万不要错误的认为创建了某类型的数组,同时就创建了N个该类型的对象。代码中,数组a中所存放的那些引用,将来所指向的必定也是A类子类的对象,因为A类自身根本就不能创建对象。

七、多维数组的length属性值是多少?

看代码

caf979fcf3da8653cf7d52cb52b324e8.png

以上代码运行,输出结果会是多少呢?一部分初学者会认为输出的结果是12,因为这个数组能够存储12个数字。但程序实际运行输出的结果却是3,这是为什么呢?就是因为,无论数组本身的维度是多少,在Java虚拟机看来,这个数组都是一维数组!有小伙伴可能不解:a明明是一个二维数组,怎么会被当作是一维数组呢?我们可以看下图来帮助理解

我们可以看到,图中是一个3行4列的二维数组。但是如果我们把数组中的每一行看作是一个元素,那么这个二维数组就可以被理解为一个一维数组,数组中包含3个元素,只不过每个元素并不是一个简单的数字,而是一个一维数组。也就是说,这个二维数组可以被理解为”由3个一维数组组成的数组”。Java语言就是以这样的方式来管理数组的,无论数组维度是多少,都被当作一维数组,如果数组中的元素又是数组(严格的表述应该是:数组元素是指向一维数组的引用),那么就形成了多维数组。因此,刚才在代码中看到的那个二维数组a,它的length属性值为3而不是12。

希望通过这七个问题,能够帮助初学者深入理解Java语言的数组。

如想系统学习Java编程,欢迎观看我在本站的视频课程。

阅读终点,创作起航,您可以撰写心得或摘录文章要点写篇博文。去创作
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
分治和递归算法是计算机科学中的两种重要算法思想,它们可以帮助我们解决很多复杂的问题。下面我介绍一下这两种算法以及它们在Java中的实现。 1. 分治算法 分治算法是指将一个大问题分成若干个小问题,然后逐个解决这些小问题,最后将所有小问题的解合并起来得到大问题的解。分治算法通常采用递归的方式来实现。 Java代码示例: ```java public int divideAndConquer(int[] nums, int left, int right){ if(left == right) return nums[left]; // 如果数组只有一个元素,则直接返回该元素 int mid = (left + right) / 2; // 将数组分成两部分 int leftSum = divideAndConquer(nums, left, mid); // 递归求解左半部分的和 int rightSum = divideAndConquer(nums, mid + 1, right); // 递归求解右半部分的和 return leftSum + rightSum; // 合并左右两部分的和 } ``` 上面的代码实现了求解一个数组中所有元素的和的问题。首先将数组分成两部分,然后递归地求解左右两部分的和,最后将左右两部分的和合并起来得到整个数组的和。 2. 递归算法 递归算法是指在函数的定义中调用自身的算法。递归算法通常用于解决具有递归结构的问题,比如树和图等数据结构。递归算法需要满足两个条件:基本情况和递归情况。 Java代码示例: ```java public int fibonacci(int n){ if(n <= 1) return n; // 基本情况 return fibonacci(n-1) + fibonacci(n-2); // 递归情况 } ``` 上面的代码实现了求解斐波那契数列的第n项的值的问题。斐波那契数列中的第一项和第二项都是1,从第三项开始,每一项都是前两项的和。递归算法中,基本情况是当n小于等于1时直接返回n,递归情况是求解前两项的和,然后递归求解前两项中的每一项。 希望这些示例代码能够帮助你理解分治和递归算法的基本思想和Java实现方式。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

Ruin-鸣

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

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

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

打赏作者

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

抵扣说明:

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

余额充值