作为程序员每天除了写很多 if else
之外,写的最多的也包含 for
循环了,都知道我们 Java
中常用的 for
循环有两种方式,一种是使用 for loop
,另一种是使用 foreach
,那如果问你,这两种方式哪一种效率最高,你的回答是什么呢?
首先我们先通过代码来实际测试一下,在计算耗时之前我们先创建一个大小集合,然后通过不断的获取集合中的内容来测试耗时。
package com.example.demo;
import java.util.ArrayList;
import java.util.List;
public class ForTest {
public static void main(String[] args) {
//获取一个指定大小的 List 集合
List<Integer> list = getList(1000000);
// 开启 for loop 耗时计算
long startFor = System.currentTimeMillis();
for (int i = 0; i < list.size(); i++) {
Integer integer = list.get(i);
}
long costFor = System.currentTimeMillis() - startFor;
System.out.println("for loop cost for ArrayList:" + costFor);
// forEach 耗时计算
long forEachStartTime = System.currentTimeMillis();
for (Integer integer : list) {
}
long forEachCost = System.currentTimeMillis() - forEachStartTime;
System.out.println("foreach cost for ArrayList:" + forEachCost);
}
public static List<Integer> getList(int size) {
List<Integer> list = new ArrayList<>();
for (int i = 0; i < size; i++) {
list.add(i);
}
return list;
}
}
简单说明一下上面的带,先创建一个 List
,然后通过两种方式的遍历来计算耗时,根据集合的大小不同,我们进行运行会得到下面的一些测试数据,不同人的机器上面运行的时间会不一定,不过差距应该也不会太大。
size= | 10000 | 100000 | 1000000 | 10000000 |
---|---|---|---|---|
for loop | 1 | 2 | 10 | 12 |
for each | 1 | 3 | 17 | 34 |
通过上面的测试结果我们可以发现,在集合相对较小的情况下,for loop
和foreach
两者的耗时基本上没有什么差别,当集合的数据量相对较大的时候,可以明显看的出来,for loop
的效率要比 foreach
的效率高。
至于为什么在大数据量的情况下 forEach
的效率要比 for
低,我们就要看下 forEach
的原理了。forEach
其实不是一种新的语法,而是一种 Java
的语法糖。在编译时,编译器会将这段代码转换成迭代器实现,并编译成字节码,我们可以再简单的看个 case
,来实际看下字节码信息。
我们再编写一个简单的类,代码如下
package com.example.demo;
import java.util.ArrayList;
import java.util.List;
/**
* <br>
* <b>Function:</b><br>
* <b>Author:</b>@author ziyou<br>
* <b>Date:</b>2022-06-26 13:06<br>
* <b>Desc:</b>无<br>
*/
public class ForEachTest {
List<Integer> list;
public void main(String[] args) {
for (Integer integer : list) {
}
}
}
通过 javac ForEachTest.java
编译成 class 文件,再通过 javap -v ForEachTest
反编译,我们就会得到下面的字节码内容:
Classfile /Users/silence/Downloads/demo/src/test/java/com/example/demo/ForEachTest.class
Last modified 2022-6-26; size 643 bytes
MD5 checksum 9cf01f7c8c87c2b4d62c39d437025b7f
Compiled from "ForEachTest.java"
public class com.example.demo.ForEachTest
minor version: 0
major version: 52
flags: ACC_PUBLIC, ACC_SUPER
Constant pool:
#1 = Methodref #8.#23 // java/lang/Object."<init>":()V
#2 = Fieldref #7.#24 // com/example/demo/ForEachTest.list:Ljava/util/List;
#3 = InterfaceMethodref #25.#26 // java/util/List.iterator:()Ljava/util/Iterator;
#4 = InterfaceMethodref #27.#28 // java/util/Iterator.hasNext:()Z
#5 = InterfaceMethodref #27.#29 // java/util/Iterator.next:()Ljava/lang/Object;
#6 = Class #30 // java/lang/Integer
#7 = Class #31 // com/example/demo/ForEachTest
#8 = Class #32 // java/lang/Object
#9 = Utf8 list
#10 = Utf8 Ljava/util/List;
#11 = Utf8 Signature
#12 = Utf8 Ljava/util/List<Ljava/lang/Integer;>;
#13 = Utf8 <init>
#14 = Utf8 ()V
#15 = Utf8 Code
#16 = Utf8 LineNumberTable
#17 = Utf8 main
#18 = Utf8 ([Ljava/lang/String;)V
#19 = Utf8 StackMapTable
#20 = Class #33 // java/util/Iterator
#21 = Utf8 SourceFile
#22 = Utf8 ForEachTest.java
#23 = NameAndType #13:#14 // "<init>":()V
#24 = NameAndType #9:#10 // list:Ljava/util/List;
#25 = Class #34 // java/util/List
#26 = NameAndType #35:#36 // iterator:()Ljava/util/Iterator;
#27 = Class #33 // java/util/Iterator
#28 = NameAndType #37:#38 // hasNext:()Z
#29 = NameAndType #39:#40 // next:()Ljava/lang/Object;
#30 = Utf8 java/lang/Integer
#31 = Utf8 com/example/demo/ForEachTest
#32 = Utf8 java/lang/Object
#33 = Utf8 java/util/Iterator
#34 = Utf8 java/util/List
#35 = Utf8 iterator
#36 = Utf8 ()Ljava/util/Iterator;
#37 = Utf8 hasNext
#38 = Utf8 ()Z
#39 = Utf8 next
#40 = Utf8 ()Ljava/lang/Object;
{
java.util.List<java.lang.Integer> list;
descriptor: Ljava/util/List;
flags:
Signature: #12 // Ljava/util/List<Ljava/lang/Integer;>;
public com.example.demo.ForEachTest();
descriptor: ()V
flags: ACC_PUBLIC
Code:
stack=1, locals=1, args_size=1
0: aload_0
1: invokespecial #1 // Method java/lang/Object."<init>":()V
4: return
LineNumberTable:
line 13: 0
public void main(java.lang.String[]);
descriptor: ([Ljava/lang/String;)V
flags: ACC_PUBLIC
Code:
stack=1, locals=4, args_size=2
0: aload_0
1: getfield #2 // Field list:Ljava/util/List;
4: invokeinterface #3, 1 // InterfaceMethod java/util/List.iterator:()Ljava/util/Iterator;
9: astore_2
10: aload_2
11: invokeinterface #4, 1 // InterfaceMethod java/util/Iterator.hasNext:()Z
16: ifeq 32
19: aload_2
20: invokeinterface #5, 1 // InterfaceMethod java/util/Iterator.next:()Ljava/lang/Object;
25: checkcast #6 // class java/lang/Integer
28: astore_3
29: goto 10
32: return
LineNumberTable:
line 17: 0
line 19: 29
line 20: 32
StackMapTable: number_of_entries = 2
frame_type = 252 /* append */
offset_delta = 10
locals = [ class java/util/Iterator ]
frame_type = 250 /* chop */
offset_delta = 21
}
SourceFile: "ForEachTest.java"
反编译的内容很多,不一一解释,可以看到这个字节码的一般含义是使用 getfield
命令获取变量,并调用 List.iterator
获取迭代器实例再调用 iterator.hasNext
,如果返回 true
,则调用 iterator.next
方法,这是迭代器遍历集合的实现逻辑。
写到这里有小伙伴就要问了,那以后遇到 List
集合我就用 for loop
了,不用foreach
了,毕竟前者的效率更好。那么接下来我们再看一个 case
,这里我们把 ArrayList
换成 LinkedList
,代码如下:
public static List<Integer> getList(int size) {
List<Integer> list = new LinkedList<>();
for (int i = 0; i < size; i++) {
list.add(i);
}
return list;
}
size= | 1000 | 10000 | 100000 |
---|---|---|---|
for loop | 27 | 129 | 7654 |
for each | 2 | 2 | 15 |
从上面的数据可以很明显的看到,当在处理 LinkedList
的时候,for loop
明显就慢很多了。相信具体的原因大家也知道,ArrayList 底层是基于数组结构的,所以使用 for loop
操作起来会很快,时间复杂度是 O(1)
,但是 LinkedList
底层是链表结构,此时如果在想通过索引来操作数据,时间复杂度将是 O (n*n)
。
所以具体使用哪种循环方式以及具体需要使用哪种数据结构,都需要根据实际的业务情况来选择,任何一种方案的存在都是合理的,你小伙你们认为呢?欢迎在评论区留言讨论。
本文由哈喽比特于2年以前收录,如有侵权请联系我们。
文章来源:https://mp.weixin.qq.com/s/TV0iyVpkzH0gN9lMXv0TGA
京东创始人刘强东和其妻子章泽天最近成为了互联网舆论关注的焦点。有关他们“移民美国”和在美国购买豪宅的传言在互联网上广泛传播。然而,京东官方通过微博发言人发布的消息澄清了这些传言,称这些言论纯属虚假信息和蓄意捏造。
日前,据博主“@超能数码君老周”爆料,国内三大运营商中国移动、中国电信和中国联通预计将集体采购百万台规模的华为Mate60系列手机。
据报道,荷兰半导体设备公司ASML正看到美国对华遏制政策的负面影响。阿斯麦(ASML)CEO彼得·温宁克在一档电视节目中分享了他对中国大陆问题以及该公司面临的出口管制和保护主义的看法。彼得曾在多个场合表达了他对出口管制以及中荷经济关系的担忧。
今年早些时候,抖音悄然上线了一款名为“青桃”的 App,Slogan 为“看见你的热爱”,根据应用介绍可知,“青桃”是一个属于年轻人的兴趣知识视频平台,由抖音官方出品的中长视频关联版本,整体风格有些类似B站。
日前,威马汽车首席数据官梅松林转发了一份“世界各国地区拥车率排行榜”,同时,他发文表示:中国汽车普及率低于非洲国家尼日利亚,每百户家庭仅17户有车。意大利世界排名第一,每十户中九户有车。
近日,一项新的研究发现,维生素 C 和 E 等抗氧化剂会激活一种机制,刺激癌症肿瘤中新血管的生长,帮助它们生长和扩散。
据媒体援引消息人士报道,苹果公司正在测试使用3D打印技术来生产其智能手表的钢质底盘。消息传出后,3D系统一度大涨超10%,不过截至周三收盘,该股涨幅回落至2%以内。
9月2日,坐拥千万粉丝的网红主播“秀才”账号被封禁,在社交媒体平台上引发热议。平台相关负责人表示,“秀才”账号违反平台相关规定,已封禁。据知情人士透露,秀才近期被举报存在违法行为,这可能是他被封禁的部分原因。据悉,“秀才”年龄39岁,是安徽省亳州市蒙城县人,抖音网红,粉丝数量超1200万。他曾被称为“中老年...
9月3日消息,亚马逊的一些股东,包括持有该公司股票的一家养老基金,日前对亚马逊、其创始人贝索斯和其董事会提起诉讼,指控他们在为 Project Kuiper 卫星星座项目购买发射服务时“违反了信义义务”。
据消息,为推广自家应用,苹果现推出了一个名为“Apps by Apple”的网站,展示了苹果为旗下产品(如 iPhone、iPad、Apple Watch、Mac 和 Apple TV)开发的各种应用程序。
特斯拉本周在美国大幅下调Model S和X售价,引发了该公司一些最坚定支持者的不满。知名特斯拉多头、未来基金(Future Fund)管理合伙人加里·布莱克发帖称,降价是一种“短期麻醉剂”,会让潜在客户等待进一步降价。
据外媒9月2日报道,荷兰半导体设备制造商阿斯麦称,尽管荷兰政府颁布的半导体设备出口管制新规9月正式生效,但该公司已获得在2023年底以前向中国运送受限制芯片制造机器的许可。
近日,根据美国证券交易委员会的文件显示,苹果卫星服务提供商 Globalstar 近期向马斯克旗下的 SpaceX 支付 6400 万美元(约 4.65 亿元人民币)。用于在 2023-2025 年期间,发射卫星,进一步扩展苹果 iPhone 系列的 SOS 卫星服务。
据报道,马斯克旗下社交平台𝕏(推特)日前调整了隐私政策,允许 𝕏 使用用户发布的信息来训练其人工智能(AI)模型。新的隐私政策将于 9 月 29 日生效。新政策规定,𝕏可能会使用所收集到的平台信息和公开可用的信息,来帮助训练 𝕏 的机器学习或人工智能模型。
9月2日,荣耀CEO赵明在采访中谈及华为手机回归时表示,替老同事们高兴,觉得手机行业,由于华为的回归,让竞争充满了更多的可能性和更多的魅力,对行业来说也是件好事。
《自然》30日发表的一篇论文报道了一个名为Swift的人工智能(AI)系统,该系统驾驶无人机的能力可在真实世界中一对一冠军赛里战胜人类对手。
近日,非营利组织纽约真菌学会(NYMS)发出警告,表示亚马逊为代表的电商平台上,充斥着各种AI生成的蘑菇觅食科普书籍,其中存在诸多错误。
社交媒体平台𝕏(原推特)新隐私政策提到:“在您同意的情况下,我们可能出于安全、安保和身份识别目的收集和使用您的生物识别信息。”
2023年德国柏林消费电子展上,各大企业都带来了最新的理念和产品,而高端化、本土化的中国产品正在不断吸引欧洲等国际市场的目光。
罗永浩日前在直播中吐槽苹果即将推出的 iPhone 新品,具体内容为:“以我对我‘子公司’的了解,我认为 iPhone 15 跟 iPhone 14 不会有什么区别的,除了序(列)号变了,这个‘不要脸’的东西,这个‘臭厨子’。