foreach循环符合就不往下走了_什么情况?为什么说写 Java 的人 for循环得用好?...

38e0fffddfc16bae38743a034d759f45.png

推荐阅读

Java程序员danni:金三即逝,连这份“Java春招手册”都没刷过,你拿什么备战银四?​zhuanlan.zhihu.com
0c2470b4bd452f3b66620d40e1adb49d.png
Java 中的循环有很多种,但是什么情况下用哪种,哪种效率高以及每种的特性,相信大多数人没有去深究过,这里面的学问可大着哩,一起来看看吧!

321db895a3cb9fcf0e537079c4233c64.png

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循环的关键字开头具有以下特点:

无需声明和初始化循环计数器变量,而是声明一个与数组的基本类型相同类型的变量,然后是冒号,然后是冒号,然后是数组名。
在循环主体中,可以使用创建的循环变量,而不是使用索引数组元素。
它通常用于遍历数组或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
作者:代码宇宙
原文链接: https:// juejin.im/post/5e8343fb e51d45470652e995
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: Java 中使用 Map.forEach循环中,需要使用 final 修饰的常量数组是因为 forEach 方法使用的是内部迭代器,该迭代器使用了闭包的概念,闭包中可能会引用到外部的变量。如果不使用 final 修饰,则外部的变量可能会被修改,而在闭包中使用的这个变量的值可能会发生变化,这会导致结果的不确定性。因此,为了保证结果的可预测性和程序的正确性,Java 中的 forEach 方法要求使用 final 修饰的常量数组。 ### 回答2: 在Java中使用map.forEach时,循环体内的变量和参数必须是final修饰的常量数组,而不能使用基本类型和包装类型的原因如下: 1. map.forEach方法使用了Lambda表达式,Lambda表达式的特点是闭包,它可以捕获外部的变量。在循环体内部,Lambda表达式可能对外部的变量进行修改,因此外部的变量必须是不可变的,即final修饰的常量。 2. 对于基本类型和包装类型的值,赋值操作实际上是将新值赋给了一个新的变量,而原来的变量仍然指向原来的值。在循环体内部对基本类型和包装类型的变量进行修改操作,相当于修改了新变量的值,而不是原来的变量指向的值。因此,如果在循环体内部修改基本类型和包装类型的值,循环体外部的值不会受到影响。 3. Java中的基本类型和包装类型都是值传递,而不是引用传递。在循环体内部对基本类型和包装类型的变量进行修改,并不会影响到循环体外部的值。因此,为了保证循环体内部对变量的修改能够生效,必须使用final修饰的常量,以保证循环体内外的变量引用的是同一个对象。 综上所述,为了确保在使用map.forEach时,循环体内部对变量的修改生效,仅能使用final修饰的常量数组。这样可以保证循环体内外的变量引用的是同一个对象,从而使得循环体内部对变量的修改能够影响到循环体外部的值。 ### 回答3: 在Java中,当我们在forEach循环内部使用局部变量时,该变量必须是final修饰的常量数组,而不能是基本类型或包装类型。 这是因为forEach循环是通过迭代器来遍历集合,而迭代器在循环中是一个匿名内部类实现的。正常情况下,局部变量在栈上分配内存,当调用方法结束后,栈帧被销毁,局部变量也随之销毁。但是对于匿名内部类来,它可能会在循环过程中被多次使用,而不像普通局部变量那样只有一次。所以,为了保证在匿名内部类中能够访问到局部变量,Java要求我们将局部变量声明为final,这样在内部类中就能够正确访问了。 但基本类型和包装类型在Java中有一个特点,即它们是值传递。这意味着,当我们将基本类型或包装类型作为参数传递给方法时,实际上传递的是其值的一个副本,而不是对原始值的引用。因此,在forEach循环内部使用final修饰的基本类型或包装类型并没有太大的实际意义,因为它们在循环中使用的是它们的副本,而不是原始值。 综上所述,为了在forEach循环中能够正确访问局部变量,我们需要将其声明为final修饰的常量数组。这样,在循环中我们使用的是该常量数组的引用,而不是对它的拷贝,这样就确保了匿名内部类能够正确访问到这个数组。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值