java 循环 效率_ArrayList哪种循环效率更好你真的清楚吗

ArrayList简介ArrayList 是一个数组队列,相当于 动态数组。与Java中的数组相比,它的容量能动态增长。它继承于AbstractList,实现了List, RandomAccess, Cloneable, java.io.Serializable这些接口。

看过ArrayList 源码的同学有没有注意过有这么一个细节:为什么ArrayList实现了RandomAccess这个接口,但是 LinkedList却没有实现这个接口?这是一个空接口,里面没有任何的方法,有什么作用呢?

答案: RandomAccess 是一个标志接口,表明实现这个这个接口的 List 集合是支持快速随机访问的。也就是说,实现了这个接口的集合是支持 快速随机访问 策略的。而LinkedList是不能实现随机访问的。

ArrayList数据结构

ArrayList包含了两个重要的对象:elementData 和 size。elementData 是"Object[]类型的数组",它保存了添加到ArrayList中的元素。实际上,elementData是个动态数组。

那是不是有人就会问既然ArrayList本质是数组,那为啥它的长度可以改变?

首先,数组的确长度不能改变。不过,ArrayList内部有一系列骚操作,大概就是它每次觉得长度不够就会 创建一个新数组,这个新数组的容量比原来多出50%,把原来的数组copy过来,然后把以前的数组销毁掉。

size 则是动态数组的实际大小。

ArrayList遍历方式第1种,普通for循环随机访问,通过索引值去遍历。

1 // 随机访问

2 List list = new ArrayList<>();

3 int size = list.size();

4 for (int i = 0; i < size; i++) {

5 value = list.get(i);

6 }第2种,通过迭代器遍历。即通过Iterator去遍历。

1 // 增强for循环

2 for (String s : list) {

3 value = s;

4 }第3种,增强for循环遍历。

1 // 迭代器遍历

2 Iterator iter = list.iterator();

3 while (iter.hasNext()) {

4 value = iter.next();

5 }第4种 forEach + lambda 循环遍历

1 list.forEach(p -> {

2 p.hashCode();

3 });

既然有4种遍历,那我们看看哪种遍历效率下面我们通过一个实验来看下这四种循环的耗时吧:

测试代码

1/**

2 * @Date: 2020/4/23

3 * @Description:

4 */

5public class ArrayListTest {

6 public static void main(String[] args) {

7 // 数据预热

8 /* List testList = createTestList(10);

9 testForEach(testList);

10 testFor(testList);

11 testRandFor(10,testList);*/

12 List integers = Arrays.asList(10, 50, 100,500,1000, 10000, 50000, 100000, 5000000, 10000000,30000000);

13 for (Integer i : integers) {

14 testRand(i);

15 }

16

17 }

18

19 private static void testRand(int size) {

20 System.out.println("-----------次数:" + size + "------------");

21 List list = createTestList(size);

22 // 随机访问通过索引值去遍历。

23 long time1 = System.nanoTime();

24 testRandFor(size, list);

25 long time2 = System.nanoTime();

26 // 增强for循环

27 testFor(list);

28 long time3 = System.nanoTime();

29 // 迭代器遍历

30 testIterator(list);

31 long time4 = System.nanoTime();

32 // forEach + lambda

33 testForEach(list);

34 long time5 = System.nanoTime();

35

36 System.out.println("随机访问\t\t" + (time2 - time1) / 1000 + " ms");

37 System.out.println("增强for遍历\t\t" + (time3 - time2) / 1000 + " ms");

38 System.out.println("迭代器遍历\t\t" + (time4 - time3) / 1000 + " ms");

39 System.out.println("forEach遍历\t\t" + (time5 - time4) / 1000 + " ms");

40 System.out.println();

41 }

42

43 private static void testRandFor(int size, List list) {

44 for (int i = 0; i < size; i++) {

45 list.get(i).hashCode();

46 }

47 }

48

49 private static void testFor(List list) {

50 for (String s : list) {

51 s.hashCode();

52 }

53 }

54

55 private static void testIterator(List list) {

56 Iterator iter = list.iterator();

57 while (iter.hasNext()) {

58 iter.next().hashCode();

59 }

60 }

61

62 private static void testForEach(List list) {

63 list.forEach(p -> {

64 p.hashCode();

65 });

66 }

67

68 public static List createTestList(int size) {

69 List list = new ArrayList<>(size);

70 for (int i = 0; i < size; i++) {

71 list.add(UUID.randomUUID().toString());

72 }

73 return list;

74 }

75}

测试数据结果如下:

1-----------次数:10------------

2随机访问 8 ms

3增强for遍历 5 ms

4迭代器遍历 2 ms

5forEach遍历 40358 ms

6

7-----------次数:50------------

8随机访问 4 ms

9增强for遍历 8 ms

10迭代器遍历 7 ms

11forEach遍历 5 ms

12

13-----------次数:100------------

14随机访问 13 ms

15增强for遍历 18 ms

16迭代器遍历 14 ms

17forEach遍历 10 ms

18

19-----------次数:500------------

20随机访问 54 ms

21增强for遍历 28 ms

22迭代器遍历 24 ms

23forEach遍历 57 ms

24

25-----------次数:1000------------

26随机访问 106 ms

27增强for遍历 56 ms

28迭代器遍历 50 ms

29forEach遍历 37 ms

30

31-----------次数:10000------------

32随机访问 1192 ms

33增强for遍历 892 ms

34迭代器遍历 861 ms

35forEach遍历 594 ms

36

37-----------次数:50000------------

38随机访问 3651 ms

39增强for遍历 2908 ms

40迭代器遍历 2563 ms

41forEach遍历 2712 ms

42

43-----------次数:100000------------

44随机访问 10693 ms

45增强for遍历 5273 ms

46迭代器遍历 9294 ms

47forEach遍历 3638 ms

48

49-----------次数:5000000------------

50随机访问 238922 ms

51增强for遍历 29914 ms

52迭代器遍历 30533 ms

53forEach遍历 28016 ms

54

55-----------次数:10000000------------

56随机访问 431047 ms

57增强for遍历 47151 ms

58迭代器遍历 46371 ms

59forEach遍历 38943 ms

60

61-----------次数:30000000------------

62随机访问 1163935 ms

63增强for遍历 137710 ms

64迭代器遍历 139211 ms

65forEach遍历 129960 ms结论:如果数据量比较少的话貌似四种循环耗时都差不多,但是随着数据量的增长会发现foreach的效率是最好的。

但是从上面我们会发现一个奇怪的现象,第一次循环的时候forEach遍历的时间是最长的尽管数据量非常少也会这样。但是后面的耗时就正常了。如果放开测试里面的预热代码,每次跑出来的耗时也是正常的。

这个结论貌似和网上的一些结论有点误差:如果你在百度上搜索java for foreach java8 等关键词会出现很多的搜索结果,比如这几个循环效率的对比。并且很多博主的结论是java8的foreach循环是真的菜,效率不是差的一点点!!!慎用,之类的。

若java8的foreach效率如此低下,为何还要推出?难道jdk的开发人员不会优化一下?带着这个思考,我仔细看了“已往之不谏”的博主最后为java8 正名的博客,写的不错,测试也很充分(说实话,没有仔细的阅读)但是结论很明显。java8胜了。作者为了证明java8不是吃素的,确实下了不少功夫。最后的最后,作者提到了,“java8的foreach预热是jvm级别的,需要预热。”原文链接感兴趣的可以去看下。

ArrayList删除数据

虽然有四种遍历方式,但是能够正确删除数据的方式只有两种第1种通过迭代器进行删除。这种方式的话,也是《阿里代码规约》所推荐的。

1 Iterator iter = list.iterator();

2 while (iter.hasNext()) {

3 iter.next().hashCode();

4 iter.remove();

5 }第2种倒序循环删除

1 for(int i = list.size()-1;i>=0;i--){

2 list.remove(i);

3 }

下面再演示下错误的删除操作普通for循环正序删除,删除过程中元素向左移动,不能删除重复的元素

1 List list = new ArrayList<>();

2 list.add("1");

3 list.add("1");

4 list.add("2");

5 for(int i=0;i

6 list.remove(i);

7 }

8 System.out.println(String.join(",",list));

结果输出:1增强for循环删除会抛出 java.util.ConcurrentModificationException

ArryList注意点谨慎使用ArrayList中的subList方法ArrayList的subList结果不可强转成ArrayList,否则会抛出ClassCastException 异常,即 java.util.RandomAccessSubList cannot be cast to java.util.ArrayList. 说明:subList 返回的是 ArrayList 的内部类 SubList,并不是 ArrayList ,而是 ArrayList 的一个视图,对于 SubList 子列表的所有操作最终会反映到原列表上。

1 List list = new ArrayList<>();

2 list.add("1");

3 list.add("1");

4 list.add("2");

5 ArrayList strings = (ArrayList)list.subList(0, 1);

6

7 运行结果:

8 Exception in thread "main" java.lang.ClassCastException: java.util.ArrayList$SubList cannot be cast to java.util.ArrayList

9 at com.workit.demo.listener.ArrayListTest.main(ArrayListTest.java:29)在 subList 场景中,高度注意对原集合元素个数的修改,会导致子列表的遍历、增加、 删除均会产ConcurrentModificationException 异常。

1 List list = new ArrayList<>();

2 list.add("1");

3 list.add("1");

4 list.add("2");

5 List subList = list.subList(0, 1);

6 // 对原List增加一个值

7 list.add("10");

8 subList.add("11"); // 这一行会报 java.util.ConcurrentModificationException初始化List的时候尽量指定它的容量大小。(尽量减少扩容次数)

结束由于自己才疏学浅,难免会有纰漏,假如你发现了错误的地方,还望留言给我指出来,我会对其加以修正。

如果你觉得文章还不错,你的转发、分享、赞赏、点赞、留言就是对我最大的鼓励。

感谢您的阅读,十分欢迎并感谢您的关注。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值