为什么说写 Java 的人 for循环得用好?

Java 中的循环有很多种,但是什么情况下用哪种,哪种效率高以及每种的特性,相信大多数人没有去深究过,这里面的学问可大着哩,一起来看看吧!

Java 循环的 4 种写法

注意,是四种写法,并不是说底层的四种实现方式,这四种写法各有千秋,但是也是最常用的几种

  1. 普通的 fori 循环
  2. forEach 语法糖
  3. lambda表达式 forEach
  4. 原生迭代器

注意,以下示例的 User 对象源码如下:

class User {
        private String name;
        private String address;
        private Integer age;

        public User(String name, String address, Integer age) {
            this.name = name;
            this.address = address;
            this.age = age;
        }

        public String getName() {
            return name;
        }

        public void setName(String name) {
            this.name = name;
        }

        public String getAddress() {
            return address;
        }

        public void setAddress(String address) {
            this.address = address;
        }

        public Integer getAge() {
            return age;
        }

        public void setAge(Integer age) {
            this.age = age;
        }
    }

普通 fori 循环

普通 for 循环原理很简单,首先获取集合的长度 userList.size(),循环体内根据循环到的下标获取对应的元素, 然后每次循环 +1,达到遍历整个集合的目的。

这种写法在以前非常的常见,现在大多使用 forEach 替代。

List<User> userList = new ArrayList<>();
userList.add(new User("同学1", "北京", 10));
userList.add(new User("同学2", "上海", 15));
userList.add(new User("同学3", "广州", 12));

// 普通 for 循环
for (int i = 0; i < userList.size(); i++) {
    User user = userList.get(i);
    System.out.println(user);
}

输出:

User{name='同学1', address='北京', age=10}
User{name='同学2', address='上海', age=15}
User{name='同学3', address='广州', age=12}
User{name='同学1', address='北京', age=10}
User{name='同学2', address='上海', age=15}
User{name='同学3', address='广州', age=12}

Process finished with exit code 0

但是普通 for 循环有两个不容忽视的优点。

第一,它在循环过程中可以轻松获取下标,比如我们想在循环中寻找符合条件的下标,那就只能使用 fori 循环,

for (int i = 0; i < userList.size(); i++) {
   User user = userList.get(i);
    if(user.age == 15){
        return i;
    }
}

第二点是它并非迭代器实现,也就是说在循环过程中它可以轻松的修改集合内的元素,增删改都没有问题,虽然不推荐这样做,但是这样的需求在实际开发中还是可能遇到。

int size = userList.size();
// 普通 for 循环
for (int i = 0; i < size; i++) {
    size = userList.size();
    User user = userList.get(i);
    if (user.age == 15) {
        userList.remove(2);
    }
}

forEach循环

For-Each 是 Java5 中引入的另一种数组遍历技术,它以类似于常规for循环的关键字开头具有以下特点:

  1. 无需声明和初始化循环计数器变量,而是声明一个与数组的基本类型相同类型的变量,然后是冒号,然后是冒号,然后是数组名。
  2. 在循环主体中,可以使用创建的循环变量,而不是使用索引数组元素。
  3. 它通常用于遍历数组或Collections类(例如ArrayList)

语法

for (type var : array) 
{ 
    statements using var;
}

示例

for (int i=0; i<arr.length; i++) 
{ 
    type var = arr[i];
    statements using var;
}

应用到 fori 的例子

for (User user : userList) {
   System.out.println(user);
}

输出

User{name='同学1', address='北京', age=10}
User{name='同学2', address='上海', age=15}
User{name='同学3', address='广州', age=12}
User{name='同学1', address='北京', age=10}
User{name='同学2', address='上海', age=15}
User{name='同学3', address='广州', age=12}

Process finished with exit code 0

局限性:

当你想要在循环体内修改数组时,for-each 循环不合适,你应该选择普通 fori 循环

for (int num : marks) 
{
    // only changes num, not the array element
    num = num*2; 
}

forEach 不跟踪索引,内部使用迭代器实现,所以我们在循环过程中没办法获取到索引

for (int num : numbers) {
    if (num == target) {
        return ???;   // do not know the index of num
    }
}
For - each only iterates forward over the array in single steps
// cannot be converted to a for-each loop
for (int i = numbers.length - 1; i > 0; i--) {
    System.out.println(numbers[i]);
}
For - each cannot process two decision making statements at once
// cannot be easily converted to a for-each loop
for (int i = 0; i < numbers.length; i++) {
    if (numbers[i] == arr[i]) { ...
    }
}

lambda 表达式 forEach

userList.forEach(e -> {
    System.out.println(e);
});

这种写法相比 forEach 更加的简单,但是存在一个很麻烦的问题,由于 lambda 是基于内部类实现的,所以我们在循环体内如果想修改外部变量,比如这样

int i = 0;
userList.forEach(e -> {
    System.out.println(e);
    i++;
});

代码中的 i++ 就会报错,因为内部类无法直接访问外部资源,Variable used in lambda expression should be final or effectively final,需要我们将变量修改为 Atomic ,如下:

AtomicInteger i = new AtomicInteger();
userList.forEach(e -> {
    System.out.println(e);
    i.set(i.getAndIncrement()+1);
});

是不是很蛋疼哩~

迭代器 iterator

迭代器在现在实际开发中使用比较少了,它长这个样子,其实 forEach 的底层就是迭代器实现。

  • forEach 中对于list编译器会调用 Iterable 接口的 iterator 方法来循环遍历数组的元素,iterator方法中是调用Iterator接口的的 next() 和 hasNext() 方法来做循环遍历。java中有一个叫做迭代器模式的设计模式,这个其实就是对迭代器模式的一个实现。

  • 对于数组,就是转化为对数组中的每一个元素的循环引用

Iterator<User> iterator = userList.iterator();
while (iterator.hasNext()) {
    System.out.println(iterator.next());
}

执行结果

User{name='同学1', address='北京', age=10}
User{name='同学2', address='上海', age=15}
User{name='同学3', address='广州', age=12}

Process finished with exit code 0

好了,关于 Java 中我了解的循环的相关内容就讲完了,如果对你有帮助,可以关注我,我会不定期发一些个人比较了解的技术内容。

ps: 本文中如果您发现错误的地方,请私信或者评论指出,感谢!

欢迎关注我的微信公众号:代码宇宙

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,658评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,482评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,213评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,395评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,487评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,523评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,525评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,300评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,753评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,048评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,223评论 1 343
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,905评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,541评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,168评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,417评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,094评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,088评论 2 352

推荐阅读更多精彩内容

  • Iterator(遍历器)的概念 JavaScript原有的表示“集合”的数据结构,主要是数组和对象,ES6又添加...
    oWSQo阅读 613评论 0 1
  • 我们在学习web前端的路程起步时总是疑问,我们如何更好的遍历元素呢?迭代器和生成器是什么?今天为大家带上与精彩的E...
    侬姝沁儿阅读 3,316评论 0 6
  • 一、Iterator(遍历器)的概念 遍历器(Iterator) 是一种接口,为各种不同的数据结构提供统一访问机制...
    了凡和纤风阅读 383评论 0 1
  • 一、Iterator(遍历器)的概念 Iterator 是一种接口,为各种不同的数据结构提供统一的访问机制。 任何...
    magic_pill阅读 314评论 0 1
  • 感觉自己的存在好像没有意义,总是给家人创造麻烦。不过我就是这样做了……死神再一次颠覆了我的人生,他鬼魅笑看...
    百鬼夜戏阅读 170评论 0 1