我国是体育大国,但什么最让咱们伤心?足球。每次世界杯都冲不进去,每次亚洲杯都差点冲不进去,很蛋疼的一种感觉。其实我看比赛发现很多时候不是因为丢球输掉比赛,是因为进不了球。德国阿根廷也丢球,但人家能进球呀,可咱们进球太难,锋无力。2017年世预赛时好不容易对韩国进了一个球,估计这辈子都忘不了那场比赛:
假如中国队锋无力的状况在里皮的调教下改善了,比如现在又到了世界杯预选赛了,中国队前五场球进球数如下:
咱先不讨论赢不赢比赛,至少这次从进球数看还不错。我的问题来了,计算一下总共进了多少球。这几个数不大,很容易计算,结果应该是12。5场进12个球,表现相当好了。那按之前学过的知识写这个程序,我们就会声明五个变量,然后给它们加起来:
你暂时没别的办法,只能这么写。可如果中国队表现好,一直进球,进了12强赛呢?然后12强赛表现也好一直进球进了世界杯呢?然后世界杯又一直赢最后拿了冠军呢?国足或许会踢很多很多场比赛,你得为每一场比赛都声明变量存进球数,0也得存。别说不可能夺冠,世上一切皆有可能,梦想要有的,万一实现了呢?万一国足笑到最后了呢?可国足笑了,你哭了,因为它踢多少场,你就得声明多少个变量。这肯定不靠谱。为了不让大家哭,java设计了一个解决方案,就是数组。数组,从名字上看,就是存储一组数的空间。更准确的说,存储一组数据的空间。学变量的时候我说变量其实就是内存中的一个空间,里面存着数据。数组呢?就是一组空间,每一个里面都存着一个数据:
注意,这些数据要么是相同类型的,要么就是可以自动转换的。我们把这些数据统称为数组元素。回到咱们这个例子,这些元素都是整型的,是可以放到一个数组里的。数组有好几种声明方法,第一种:
用中括号来表示数组,表示不再只存一个值或元素。比如我写int[] arr = {3, 2, 1, 4, 2},表示arr这个数组里面存放了五个整型数据。数组声明完之后通过数组的下标来访问元素,格式是:
注意,数组下标是从0开始的,arr[0]代表第一个元素,所以例子中五个数依次为arr[0] = 3,arr[1] = 2,arr[2] = 1,arr[3] = 4, arr[4] = 2。我们也可以用for循环来遍历它,然后把遍历出来的值都加上,程序修改如下:
这段程序应该不难看懂,从i=0开始循环,每次打印出一个arr[i]的值。声明出来一个叫total的变量存储总进球数,把每次循环的结果加进去。String.valueOf()用来把场次转型成字符串以便打印,i的初始值是0,所以一开始用i+1。把这个for循环记住,遍历数组就个方式。记得我们介绍循环时说for循环还有一种特殊形式,我们在这块儿把这个小内容加进去。注意看我改一下程序:
for循环括号里的内容被一个冒号分成了两个部分,前面是变量i,没变化;后面的是数组本身。这句代码就是按顺序遍历这个数组的下标的意思。因为数组的长度是已知的,我们不用设置终值;又因为按顺序遍历数组每个下标,我们也不用设置步长。这里用for循环的特殊形式就显得很方便。当然,用这个格式的前提就是你得有个能装变量的东西,数组可以满足这个要求,以后我们会介绍容器,也可以满足要求。
那如果我访问arr[5]呢?会报错,因为数组越界了。本来嘛,声明的空间里总共就有五个格,arr[5]指向的是第六个格,你非要访问一个不存在的不等报错等啥呢?那有人说如果中国队确实踢了很多很多场比赛,数组里元素太多我数不过来,有没有方法查看数组的大小?有,用arr.length()的方法。有人说这个看着怎么那么像对象调用方法啊?没错,其实数组本身就是一个引用类型。有人说你瞎说,刚才例子int arr[] = {3, 2, 1, 4, 2}中元素不都是整型的吗?哪儿来的引用类型?注意,整型指的是数组中的元素,并不是数组本身,也就是说这个int只不过指明了该数组要存放什么类型的元素而已。而arr本身因为带个中括号,所以是数组型,属于引用类型,存的是整个数组的地址,其实精确点说是数组中第一个元素的地址。因为数组是连成一串的,有了第一个元素的地址就能定位整个的:
这是个整型数组的例子。比如我们还可以为对手也建个数组:String arr_duishou[] = {"伊朗", "韩国", "澳大利亚", "日本"};这里面arr_duishou也是引用类型,它存的是第一个元素"伊朗"的地址。而String是数组的元素类型,指明数组里要存放字符串类型的元素。
回到刚才说的,在元素很多的情况下可以用arr.length作为终值遍历,程序改为:
在java中我们必须要给数组提前定好大小,而且一旦定下就不能更改了,始终定长。第一种声明方法虽然在定义数组的同时也完成了初始化,但却没有直接声明大小。第二种方法可以直接声明大小,格式:
所以,例子中我可以把声明过程写成:
其实我们还可以把中括号放在数组名后面,写成这样:
那刚才说了,数据元素要么是相同类型的,要么可以自动转换的。看我再声明一个数组:
数组的元素类型定义为float型,但元素里有一个是整型。遍历打印没问题,因为int是可以自动转成float的。但如果再加一个3.3:
不等执行就该报错啦,因为3.3不写f代表double类型,是不能自动转换的。之前学类型转换的时候讲过,自动转化只能是由低到高,你要非由高到低就得强制转换。很可惜,数组赋值时不能强制转换。
除了int , float, string这种基本类型元素,还能不能放别的呢?我们已经学了面向对象,学了类,知道了引用类型,那可不可以放引用类型元素呢?比如对象呢?答案是可以。还记得咱们之前讲类的时候那个网红的例子么?我们有王思聪,也有咪蒙,现在咱们再加一个大幂幂吧,杨幂。大幂幂其实又是明星,又是网红,流量之王。Wanghong类有三个成员变量,分别是姓名、年龄、爱好。我们可以创建一个数组把这些对象都装进去。既然现在元素类型是Wanghong这个引用类型,那声明就应该是这样的:
元素类型不再是基本类型了,所以数组里每一个格子里存的其实都是一个对象的引用,也就是对象的地址:
对于int float那些基本类型直接赋值就行,可对于引用类型来说,赋值也就是实例化对象。搞清楚这些后开始写程序,先把数组声明和Wanghong类写好。Eclipse中新建java项目WanghongArray -> 包com.test -> Wanghong.java和Test.java:
Wanghong.java内容如下,还是一个get/set封装的过程:
主类Test.java放的是Wanghong对象实例化过程:
Wanghong类中有name,age,hobby三个成员变量,变量声明成私有,为了保护;方法公有,目的为了以后给变量赋值。这是介绍封装那篇时说的,轻车熟路,没问题。而且我们还用了属性的get/set方法准备一会儿给变量赋值。赋值过程如下:
刚创建出来之后w[0],w[1],w[2]是空的,我用for循环为每一个位置赋值:比如对于w[0],实例化,就是w[0] = new Wanghong(),开辟了一个空间后w[0]里面就有了对象地址了。然后用get/set方法为对象中每一个成员变量再赋值。
最后循环打印结果,Test.java的完整代码如下:
这就是对于引用类型数组的示例。另外,数组中除了一维数组还有二维数组。二维数组是这么声明的:
声明中有两个中括号,指的分别是数组整个的行数和列数。比如我有个两行三列的二维数组,声明语句就是
二维数组下标也是从0开始,它在内存中的感觉是这样的:
和一维数组一样,给二维数组赋值也需要一个一个位置来:arr[0][0] = 1,arr[0][1] = 2, arr[0][2] = 3, arr[1][0] = 4,arr[1][1] = 5,arr[1][2] = 6。打印出来所有的二维数组元素还得是遍历。既然是二维的,那我们遍历二维数组时一个for循环够么?不够,得用两个for循环嵌套。一个控制行,一个控制列。看我写完就知道为什么了。Eclipse中新建java项目TwoDimentionalArray -> 包com.test -> Test.java:
外循环for控制行,因为有两行,所以执行两遍;内循环控制列,每次读取一行时都会按列打印arr[i][j]。不太懂嵌套循环的请复习流控机制那一节。以后用selenium写自动化代码时还有机会接触二维数组,到时我会带着大家再体会二维数组的使用方法。但是数组也有缺陷,java中的数组只允许我们要么存相同类型的数据,要么存能够自动转换的数据,除此之外都不行了。那有人又说了,我就想存不同类型的行不行?那就不能用数组了,java给我们提供了另一种数据结构来实现这个目的,叫做集合。下篇我们一起讨论。
这篇文章的源代码是WanghongArray和TwoDimentionalArray项目。
本篇知识点及注意事项:
1. 数组下标是从0开始的。
2. 在java中我们必须要给数组提前定好大小,而且一旦定下就不能更改了,始终定长。
3. 数组的类型是数组型,是引用类型。而数组的元素类型可以是任意类型,但所有元素必须类型一致。
4. java中的数组只允许我们要么存相同类型的数据,要么存能够自动转换的数据。