大家好!我是小米,一个29岁、乐于分享技术的小米。今天我们来聊聊一个经常出现在社招面试中的问题——Array 和 ArrayList 有何区别?
当时我刚跳槽到新公司,在面试时,面试官突然抛出一个看似简单的问题:“能说说 Array 和 ArrayList 的区别吗?”表面看,这问题容易得像开水白菜,但实际上,它藏着大招。
面试场景回放
面试官一脸严肃地盯着我,说:“Array 和 ArrayList 的区别不仅仅是理论知识,更体现在实际开发的运用中。”
我点点头,快速在脑海中梳理了一遍,回答道:
定义和存储方式
Array(数组):
数组是 Java 中最基本的数据结构之一,它是一个固定大小的容器,用来存储相同数据类型的元素。数组的长度在创建时就确定了,后续无法更改。
图片
ArrayList:
ArrayList 是 Java 提供的一个动态数组类,它可以随着数据量的增加自动扩容,且属于 Java Collections Framework。其底层实际上是用数组实现的。
图片
面试官点了点头,追问:它们存储数据时有什么不同?
我继续回答:数组是直接存储数据,而 ArrayList 内部通过包装类来存储对象。也就是说,ArrayList 需要装箱和拆箱,会有一定的性能开销。
大小是否可变
Array:数组的大小一旦初始化就不能改变。如果需要更大的存储空间,必须重新创建一个更大的数组,并将原有数据复制过去。
图片
ArrayList:大小是动态变化的,它会根据需要自动扩容,开发者无需手动管理。
图片
性能比较
- 数组性能更高:数组在访问和操作时性能更高,因为它是直接操作内存中的元素,没有额外的开销。
- ArrayList 灵活但略慢:ArrayList 的操作需要频繁地装箱和拆箱,尤其是当扩容时,它需要创建一个新数组并将数据复制过去,性能会受到影响。
支持的操作
Array:
数组是相对“原始”的数据结构,支持索引访问,但不支持增删操作。
ArrayList:
ArrayList 提供了丰富的操作方法,比如 add()、remove() 和 contains() 等,非常方便。
图片
是否支持泛型
Array:数组在定义时支持泛型,但是泛型数组的初始化有一定的局限性,例如不能直接创建泛型数组。
图片
ArrayList:ArrayList 是 Collections Framework 的一部分,支持泛型,使用更为灵活。
图片
多线程支持
- Array:数组本身是线程安全的,因为它是最基础的数据结构,没有额外的线程同步机制。
- ArrayList:ArrayList 不是线程安全的。如果需要在多线程环境中使用,可以用 Collections.synchronizedList() 包装它,或者直接使用线程安全的 CopyOnWriteArrayList。
真实场景下的选择
在实际开发中,我们如何选择呢?
- 需要高性能、固定大小的场景:优先选择数组,比如处理大量的数学计算或内存敏感的任务。
- 需要灵活操作或动态扩展:ArrayList 是更好的选择,比如存储用户输入的数据或动态生成的数据。
写代码小贴士
记住,Array 和 ArrayList 都有自己的适用场景。在面试中,除了理论讲清楚,还可以举一些实际的例子来说明它们的用法。最后,别忘了提到扩展知识点,比如 LinkedList 和 CopyOnWriteArrayList,这样面试官一定会对你刮目相看!