教妹学 Java之数组

开发 后端
“哥,我看你之前的文章里提到,ArrayList 的内部是用数组实现的,我就对数组非常感兴趣,想深入地了解一下,今天终于到这个环节了,好期待呀!”三妹的语气里显得很兴奋。

[[398498]]

 “哥,我看你之前的文章里提到,ArrayList 的内部是用数组实现的,我就对数组非常感兴趣,想深入地了解一下,今天终于到这个环节了,好期待呀!”三妹的语气里显得很兴奋。

“的确是的,看 ArrayList 的源码就一清二楚了。”我一边说,一边打开 Intellij IDEA,并找到了 ArrayList 的源码。

  1. /** 
  2.  * The array buffer into which the elements of the ArrayList are stored. 
  3.  * The capacity of the ArrayList is the length of this array buffer. Any 
  4.  * empty ArrayList with elementData == DEFAULTCAPACITY_EMPTY_ELEMENTDATA 
  5.  * will be expanded to DEFAULT_CAPACITY when the first element is added. 
  6.  */ 
  7. transient Object[] elementData; // non-private to simplify nested class access 
  8.  
  9. /** 
  10.  * The size of the ArrayList (the number of elements it contains). 
  11.  * 
  12.  * @serial 
  13.  */ 
  14. private int size

“瞧见没?Object[] elementData 就是数组。”我指着显示屏上这串代码继续说。

数组是一个对象,它包含了一组固定数量的元素,并且这些元素的类型是相同的。数组会按照索引的方式将元素放在指定的位置上,意味着我们可以通过索引来访问这些元素。在 Java 中,索引是从 0 开始的。

“哥,能说一下为什么索引从 0 开始吗?”三妹突然这个话题很感兴趣。

“哦,Java 是基于 C/C++ 语言实现的,而 C 语言的下标是从 0 开始的,所以 Java 就继承了这个良好的传统习惯。C语言有一个很重要概念,叫做指针,它实际上是一个偏移量,距离开始位置的偏移量,第一个元素就在开始的位置,它的偏移量就为 0,所以索引就为 0。”此刻,我很自信。

“此外,还有另外一种说法。早期的计算机资源比较匮乏,0 作为起始下标相比较于 1 作为起始下标,编译的效率更高。”

“哦。”三妹意味深长地点了点头。

我们可以将数组理解为一个个整齐排列的单元格,每个单元格里面存放着一个元素。

数组元素的类型可以是基本数据类型(比如说 int、double),也可以是引用数据类型(比如说 String),包括自定义类型。

数组的声明方式分两种。

先来看第一种:

  1. int[] anArray; 

再来看第二种:

  1. int anOtherArray[]; 

不同之处就在于中括号的位置,是跟在类型关键字的后面,还是跟在变量的名称的后面。前一种的使用频率更高一些,像 ArrayList 的源码中就用了第一种方式。

同样的,数组的初始化方式也有多种,最常见的是:

  1. int[] anArray = new int[10]; 

看到了没?上面这行代码中使用了 new 关键字,这就意味着数组的确是一个对象,只有对象的创建才会用到 new 关键字,基本数据类型是不用的。然后,我们需要在方括号中指定数组的长度。

这时候,数组中的每个元素都会被初始化为默认值,int 类型的就为 0,Object 类型的就为 null。不同数据类型的默认值不同,可以参照之前的文章。

另外,还可以使用大括号的方式,直接初始化数组中的元素:

  1. int anOtherArray[] = new int[] {1, 2, 3, 4, 5}; 

这时候,数组的元素分别是 1、2、3、4、5,索引依次是 0、1、2、3、4,长度是 5。

“哥,怎么访问数组呢?”三妹及时地插话到。

前面提到过,可以通过索引来访问数组的元素,就像下面这样:

  1. anArray[0] = 10; 

变量名,加上中括号,加上元素的索引,就可以访问到数组,通过“=”操作符可以对元素进行赋值。

如果索引的值超出了数组的界限,就会抛出 ArrayIndexOutOfBoundException。

既然数组的索引是从 0 开始,那就是到数组的 length - 1 结束,不要使用超出这个范围内的索引访问数组,就不会抛出数组越界的异常了。

当数组的元素非常多的时候,逐个访问数组就太辛苦了,所以需要通过遍历的方式。

第一种,使用 for 循环:

  1. int anOtherArray[] = new int[] {1, 2, 3, 4, 5}; 
  2. for (int i = 0; i < anOtherArray.length; i++) { 
  3.     System.out.println(anOtherArray[i]); 

通过 length 属性获取到数组的长度,然后从 0 开始遍历,就得到了数组的所有元素。

第二种,使用 for-each 循环:

  1. for (int element : anOtherArray) { 
  2.     System.out.println(element); 

如果不需要关心索引的话(意味着不需要修改数组的某个元素),使用 for-each 遍历更简洁一些。当然,也可以使用 while 和 do-while 循环。

在 Java 中,可变参数用于将任意数量的参数传递给方法,来看 varargsMethod() 方法:

  1. void varargsMethod(String... varargs) {} 

该方法可以接收任意数量的字符串参数,可以是 0 个或者 N 个,本质上,可变参数就是通过数组实现的。为了证明这一点,我们可以看一下反编译一后的字节码:

  1. public class VarargsDemo 
  2.  
  3.     public VarargsDemo() 
  4.     { 
  5.     } 
  6.  
  7.     transient void varargsMethod(String as[]) 
  8.     { 
  9.     } 

所以,我们其实可以直接将数组作为参数传递给该方法:

  1. VarargsDemo demo = new VarargsDemo(); 
  2. String[] anArray = new String[] {"沉默王二""一枚有趣的程序员"}; 
  3. demo.varargsMethod(anArray); 

也可以直接传递多个字符串,通过逗号隔开的方式:

  1. demo.varargsMethod("沉默王二""一枚有趣的程序员"); 

在 Java 中,数组与 List 关系非常密切。List 封装了很多常用的方法,方便我们对集合进行一些操作,而如果直接操作数组的话,有很多不便,因为数组本身没有提供这些封装好的操作,所以有时候我们需要把数组转成 List。

“怎么转呢?”三妹问到。

最原始的方式,就是通过遍历数组的方式,一个个将数组添加到 List 中。

  1. int[] anArray = new int[] {1, 2, 3, 4, 5}; 
  2.  
  3. List<Integer> aList = new ArrayList<>(); 
  4. for (int element : anArray) { 
  5.     aList.add(element); 

更优雅的方式是通过 Arrays 类的 asList() 方法:

  1. List<Integer> aList = Arrays.asList(anArray); 

但需要注意的是,该方法返回的 ArrayList 并不是 java.util.ArrayList,它其实是 Arrays 类的一个内部类:

  1. private static class ArrayList<E> extends AbstractList<E> 
  2.         implements RandomAccess, java.io.Serializable{} 

如果需要添加元素或者删除元素的话,需要把它转成 java.util.ArrayList。

  1. new ArrayList<>(Arrays.asList(anArray)); 

Java 8 新增了 Stream 流的概念,这就意味着我们也可以将数组转成 Stream 进行操作。

  1. String[] anArray = new String[] {"沉默王二""一枚有趣的程序员""好好珍重他"}; 
  2. Stream<String> aStream = Arrays.stream(anArray); 

如果想对数组进行排序的话,可以使用 Arrays 类提供的 sort() 方法。

基本数据类型按照升序排列

实现了 Comparable 接口的对象按照 compareTo() 的排序

来看第一个例子:

  1. int[] anArray = new int[] {5, 2, 1, 4, 8}; 
  2. Arrays.sort(anArray); 

排序后的结果如下所示:

  1. [1, 2, 4, 5, 8] 

来看第二个例子:

  1. String[] yetAnotherArray = new String[] {"A""E""Z""B""C"}; 
  2. Arrays.sort(yetAnotherArray, 1, 3, 
  3.                 Comparator.comparing(String::toString).reversed()); 

只对 1-3 位置上的元素进行反序,所以结果如下所示:

  1. [A, Z, E, B, C] 

有时候,我们需要从数组中查找某个具体的元素,最直接的方式就是通过遍历的方式:

  1. int[] anArray = new int[] {5, 2, 1, 4, 8}; 
  2. for (int i = 0; i < anArray.length; i++) { 
  3.     if (anArray[i] == 4) { 
  4.         System.out.println("找到了 " + i); 
  5.         break; 
  6.     } 

上例中从数组中查询元素 4,找到后通过 break 关键字退出循环。

如果数组提前进行了排序,就可以使用二分查找法,这样效率就会更高一些。Arrays.binarySearch() 方法可供我们使用,它需要传递一个数组,和要查找的元素。

  1. int[] anArray = new int[] {1, 2, 3, 4, 5}; 
  2. int index = Arrays.binarySearch(anArray, 4); 

“除了一维数组,还有二维数组,三妹你可以去研究下,比如说用二维数组打印一下杨辉三角。”说完,我就去阳台上休息了,留三妹在那里学习,不能打扰她。

本文转载自微信公众号「沉默王二」,可以通过以下二维码关注。转载本文请联系沉默王二公众号。

 

 

责任编辑:武晓燕 来源: 沉默王二
相关推荐

2021-06-06 20:56:48

Java内存 intern

2020-10-26 09:36:45

Java变量数据

2020-11-18 09:44:49

Java命名约定

2021-07-08 22:43:41

ThrowThrowsJava

2021-07-03 17:53:52

Java异常处理机制

2021-07-26 17:22:02

Java

2021-06-11 18:08:00

Java字符串拼接

2020-10-29 10:28:31

Java数据类型

2021-07-30 09:32:55

JavaEquals

2020-11-13 10:29:37

流程控制语句

2022-11-26 00:34:57

数组Java程序

2020-12-09 05:19:35

static关键字Java

2021-05-14 07:53:32

Java打印数组

2022-10-31 07:33:05

Javafor循环

2022-02-16 08:26:18

Go并发信号量

2018-03-06 10:46:42

代码Python买水果

2022-09-30 07:32:48

循环while循环体

2022-10-30 10:14:43

Java循环语句

2022-08-05 12:05:07

JavaGuava集合

2017-01-05 15:13:03

Java数组算法解释
点赞
收藏

51CTO技术栈公众号