程序员的进阶课-架构师之路 - 数组
从这一节开始,我们就要正式进去数据结构的世界了,那么第一个是什么呢,就是我们的数组。
在我想写数组的时候,我的第一印象是去看它的源码,很可惜,数组的实现太特殊了,找了很久,我没有找到它的源码,带着这样的思考,我就开始了Java中数组的挖掘。Wow,真香!
一、Java中数组的介绍
数组是一种最简单的复合数据类型,它是有序数据的集合,数组中的每个元素具有相同的数据类型,可以用一个统一的数组名和不同的下标来唯一确定数组中的元素。根据数组的维度,可以将其分为一维数组、二维数组和多维数组等。一定要注意,数组只能存放同一种数据类型(Object类型数组除外)。
二、数组是一个引用类型吗?
先给答案,是的,没有任何疑问。
注意,数组也是一种数据类型,它本身是一种引用类型。
数组是一种大小固定的数据结构,对线性表的所有操作都可以通过数组来实现。虽然数组一旦创建之后,它的大小就无法改变了,但是当数组不能再存储线性表中的新元素时,我们可以创建一个新的大的数组来替换当前数组。这样就可以使用数组实现动态的数据结构。
如何验证?
定义一个数组,发现它拥有Object类的所有方法。
根据这个例子,其实大家已经看出来了,数组拥有超类Object的所有方法,说明他也是一个类。并且他拥有自己的clone()方法和length属性。
三、如何了解数组的底层实现
既然数组拥有Object的所有方法,那我们是否能查看一下数组的源码,来了解一下数组的实现呢?
可惜,数组太特殊了,他的实现是虚拟机编译的时候动态生成的,所以我们无法直接查看源码,只能通过查看编译后的class的字节码一探究竟。
JVM 中数组对象是一种特殊的对象,虚拟机从数组的元数据中无法确认数组的大小,它的Object Header 比普通对象多了一个word 来存储数组的长度,length 会编译成对应的字节码读取这个field 就可以了。
我分别定义基本数据类型和引用类型来查看一下最终生成的字节码有何区别。
Object[] o = new String[11]; o[0]="1aaa"; int i=o.length; Integer[] a=new Integer[11]; a[0]=100; int j=a.length; int[] b=new int[11]; b[0]=100; int k=b.length; }
对应的字节码为:
2 anewarray #12 <java/lang/String> //anewarray代表对象数组 5 astore_1 6 aload_1 7 iconst_0 8 ldc #25 <1aaa> 10 aastore 11 aload_1 12 arraylength //arraylength代表长度 13 istore_2 14 bipush 11 16 anewarray #26 <java/lang/Integer> //anewarray代表包装类数组 19 astore_3 20 aload_3 21 iconst_0 22 bipush 100 24 invokestatic #27 <java/lang/Integer.valueOf> 27 aastore 28 aload_3 29 arraylength 30 istore 4 32 bipush 11 34 newarray 10 (int) //newarray代表基本数组类型数组 36 astore 5 38 aload 5 40 iconst_0 41 bipush 100 43 iastore 44 aload 5 46 arraylength 47 istore 6 49 return
注意:定义并初始化一个数组后,在内存中分配了两个空间,一个用于存放数组的引用变量,另一个用于存放数组本身。
进行程序开发时,要深入底层的运行机制。
看待一个数组时,一定要把数组看成两个部分:一部分是数组引用,也就是在代码中定义的数组引用变量;还有一部分是实际的数组对象,这部分是在对内存里运行的,通常无法直接访问它,只能通过数组引用变量来访问。
四、Array 的 length 域相关
在很多的资料中都写了,Array中有类似public final int length的成员变量。但是在《Java Language Specifications》10.1. Array Types中明确写了,length不是类型的一部分;
An array's length is not part of its type.
String[] s = new String[2]; System.out.println(s.length); System.out.println(s.getClass().getDeclaredFields().length); try { System.out.println(s.getClass().getDeclaredField("length")); } catch (NoSuchFieldException e) { System.out.println(e.toString()); } }
可以看到length并不是Array的成员变量。
五、Java语言规范关于Array的定义
数组在Java里是一种特殊类型,有别于普通的“类的实例”的对象。
10.1. Array Types
10.8. Class Objects for Arrays
Every array has an associated Class object, shared with all other arrays with the same component type.Although an array type is not a class, the Class object of every array acts as if:
1、The direct superclass of every array type is Object.
2、Every array type implements the interfaces Cloneable and java.io.Serializable.
数组类型是由JVM从元素类型合成出来的。
10.7. Array Members
The members of an array type are all of the following:
1、The public final field length, which contains the number of components of the array. length may be positive or zero.
从Java语言到Class文件,Java源码编译器会识别出对数组类型的length字段的访问,并生成对应的字节码。
以OpenJDK8的javac为例:
jdk8u/jdk8u/langtools: 84eb51777733 src/share/classes/com/sun/tools/javac/jvm/Gen.java
if (sym == syms.lengthVar) { code.emitop0(arraylength); result = items.makeStackItem(syms.intType); }
六、数据应用场景
这种数据结构使用一段连续的空间来存贮元素,所以可以直接通过索引来获取到某个元素,而且可以通过对元素的内容进行排序,然后使用二分法查找,从而提供查找效率。其适合的场合主要是:
1、不会频繁增删元素的场合,因为增删元素都牵涉到元素空间的重新分配,频繁的内存分配操作会大幅降低操作效率。但添加操作时,可以通过预分配足够的空间来优化添加时的效率。
2、属于随机迭代器,可以随机访问任意元素。对于已排序的元素查找起来效率较高。
七、数组总结
在看数组的时候,因为class是动态创建的,所以看了很久,但是根据数组的特性,基本可以认为数组的域和方法,类似于:
public T[] clone() { try { return (T[]) super.clone(); } catch (CloneNotSupportedException e) { throw new InternalError(e.getMessage()); } } }
数组可以是一维数组、二维数组或多维数组。
数值数组元素的默认值为 0,而引用元素的默认值为 null。
交错数组是数组的数组,因此,它的元素是引用类型,初始化为 null。交错数组元素的维度和大小可以不同。
数组的索引从 0 开始,如果数组有 n 个元素,那么数组的索引是从 0 到(n-1)。
数组元素可以是任何类型,包括数组类型。
数组类型是从抽象基类 Array 派生的引用类型。

低调大师中文资讯倾力打造互联网数据资讯、行业资源、电子商务、移动互联网、网络营销平台。
持续更新报道IT业界、互联网、市场资讯、驱动更新,是最及时权威的产业资讯及硬件资讯报道平台。
转载内容版权归作者及来源网站所有,本站原创内容转载请注明来源。
- 上一篇
x-easypdf v1.1.0 发布
x-easypdf基于pdfbox构建而来,极大降低使用门槛,以组件化的形式进行pdf的构建。简单、易用,3分钟即可上手,人人都是pdf的构建高手 本次更新如下: 1. 由于与现在软件名重名,故变更项目名(原为xpdf) 2. 统一源文件前缀为XEasyPdf 3. 新增水印组件 4. 新增图片组件 5. 新增模板填充 由于表格组件遇到些许问题,本次更新暂不提供,将在下个版本进行更新,感谢大家支持,谢谢
- 下一篇
「OSCHINA 开源软件趋势榜」即将上线,等你来揭榜
你觉得哪些项目可以进入候选名单?目前用户提名通道已经开启,具体方式见本文“如何参与”部分。 截至目前,OSCHINA 社区已收录超过 5 万款开源软件。每时每地,都有各类软件不断地被开发、被开源、被使用。但更多时候,由于传播渠道的缺乏等各类因素,一些优质项目难以被发掘,难以得到更好的建设。其中不乏一些新兴软件,它们也许有着更大的成长空间、更高的价值潜力。那么,如何打破这种局面?应当建立一种怎样的机制,使这些软件的价值不再被埋没? 「OSCHINA 开源软件趋势榜」正是本着这种精神诞生。接下来的每三个月,都会有这样一份承载着价值潜力的开源软件榜单呈现在各位眼前。同时,它不仅仅是一份榜单,也是一份针对当下的趋势反映。譬如,近两个月来出现了不少疫情相关的开源软件;此前某种语言突然受到关注时,也会有一些相关软件扎堆冒出,这些都值得被梳理和记录。近期开源软件与什么主题或事件有关?流行采用哪些技术?哪一类新软件更容易被接受?……它们或许都能在榜单中得到体现。 我们将从 2020 年第一季度开始,整理、筛选当季被收录进 OSCHINA 社区的开源软件,并从中选出最具有价值潜力的部分,最终形成「OSC...
相关文章
文章评论
共有0条评论来说两句吧...
文章二维码
点击排行
推荐阅读
最新文章
- SpringBoot2整合Thymeleaf,官方推荐html解决方案
- Hadoop3单机部署,实现最简伪集群
- Springboot2将连接池hikari替换为druid,体验最强大的数据库连接池
- Docker安装Oracle12C,快速搭建Oracle学习环境
- CentOS8安装MyCat,轻松搞定数据库的读写分离、垂直分库、水平分库
- CentOS7设置SWAP分区,小内存服务器的救世主
- SpringBoot2整合Redis,开启缓存,提高访问速度
- CentOS8,CentOS7,CentOS6编译安装Redis5.0.7
- CentOS7,CentOS8安装Elasticsearch6.8.6
- SpringBoot2整合MyBatis,连接MySql数据库做增删改查操作