一 数据元素在内存中存放方式
- 链式存储
每一个数据元素,在内存中都不要求处于相邻的位置,每个数据元素包含它下一个元素的内存地址。不可以根据元素的位置直接计算出内存地址,只能按顺序读取元素。读取一个特定位置元素的平均时间复杂度为O(n)。主要以链表为代表。Java中以LinkedList为代表。 - 顺序存储
相邻的数据元素存放于相邻的内存地址中,整块内存地址是连续的。可以根据元素的位置直接计算出内存地址,直接进行读取。读取一个特定位置元素的平均时间复杂度为O(1)。正常来说,只有基于数组实现的集合,才有这种特性。Java中以ArrayList为代表。
二 常用的遍历方法
1.通过for循环
for(int i = 0 ;i<list.size();i++) {
int j= list.get(i);
System.out.println(j);
}
遍历者自己在集合外部维护一个计数器,然后依次读取每一个位置的元素,当读取到最后一个元素后,停止。主要就是需要按元素的位置来读取元素。
遍历效率
- 链式存储:读取特定位置元素的平均时间复杂度是O(n),所以遍历整个集合的平均时间复杂度为O(n^2)
- 顺序存储:读取特定位置元素的平均时间复杂度是O(1),所以遍历整个集合的平均时间复杂度为O(n)
2.通过Iterator迭代器
Iterator it = list.iterator();
while(it.hasNext()) {
Object obj = it.next();
}
基于顺序存储集合的Iterator可以直接按位置访问数据。而基于链式存储集合的Iterator,都是需要保存当前遍历的位置。然后根据当前位置来向前或者向后移动指针。
遍历效率
- 链式存储:由于Iterator内部维护了当前遍历元素的位置,遍历下一元素只需向后移动一位指针,遍历整个集合的时间复杂度为O(n);
- 顺序存储:由于迭代器的额外操作,反而花费了更多时间
3.通过foreach循环
for (Object object : list) {
System.out.println(object);
}
foreach内部也是采用了Iterator的方式实现,只不过Java编译器帮我们生成了这些代码。
三 遍历建议
Java数据集合框架中,提供了一个RandomAccess接口,该接口没有方法,只是一个标记。通常被List接口的实现使用,用来标记该List的实现是否支持Random Access。
-
一个数据集合实现了该接口,就意味着它支持Random Access,按位置读取元素的平均时间复杂度为O(1),比如ArrayList。建议使用for循环遍历
-
若没有实现Random Access接口,比如LinkedList,建议使用迭代器遍历。