竟然有人不知道for与foreach的区别?


文章来源:https://c1n.cn/wAebg

目录

  • 问题

  • 原因

  • 总结

问题

之前有一个同事突然我问了我一个问题,说在 foreach 当中能不能删除 list 里面的元素,我当时大概说了一下是否能删除,以及原因;接下来我们来探讨一下是否能够如此。

| 遍历元素

首先,我们一一段代码为例:

String[] array = {"1", "2", "3"};
for (String i : array) {
    System.out.println(i);
}

ArrayList<String> list = new ArrayList<>();
list.add("111");
list.add("222");
list.add("333");
for (String i : list) {
    System.out.println(i);
}

遍历后结果如下:

1
2
3
111
222
333

结果毫无疑问。我们再来看看编译后的源码(idea 自带,在 target 包里打开你的类源码文件即可):

String[] array = new String[]{"1", "2", "3"};
String[] var2 = array;
int var3 = array.length;

for(int var4 = 0; var4 < var3; ++var4) {
    String i = var2[var4];
    System.out.println(i);
}

ArrayList<String> list = new ArrayList();
list.add("111");
list.add("222");
list.add("333");
Iterator var7 = list.iterator();

while(var7.hasNext()) {
    String i = (String)var7.next();
    System.out.println(i);
}

可见,遍历数组使用的是原始 for 循环,集合的话使用的是 Iterator 迭代器。

| 删除元素

哦的 k!接下来我们来删除元素:

使用 for 循环:
ArrayList<String> list = new ArrayList<>();
list.add("111");
list.add("222");
list.add("333");
log.info(list.toString());
for (int i = 0; i <list.size(); i++) {
    list.remove("222");
}

log.info(list.toString());

结果:

11:11:52.532 [main] INFO com.xiaolinge.com.hello.HelloWord - [111, 222, 333]
11:11:52.539 [main] INFO com.xiaolinge.com.hello.HelloWord - [111, 333]
显然成功!
使用 foreach:
ArrayList<String> list = new ArrayList<>();
list.add("111");
list.add("222");
list.add("333");
log.info(list.toString());
for (String i : list) {
  list.remove("222");
}
log.info(list.toString());

结果:

11:50:48.333 [main] INFO com.xiaolinge.com.hello.HelloWord - [111, 222, 333]
Exception in thread "main" java.util.ConcurrentModificationException
 at java.util.ArrayList$Itr.checkForComodification(ArrayList.java:909)
 at java.util.ArrayList$Itr.next(ArrayList.java:859)
 at com.xiaolinge.com.hello.HelloWord.main(HelloWord.java:30)

显然木有成功!

原因

迭代器内部的每次遍历都会记录 List 内部的 modcount 当做预期值,然后在每次循环中用预期值与 List 的成员变量 modCount 作比较。

但是普通 list.remove 调用的是 List 的 remove,这时 modcount++,但是 iterator 内记录的预期值=并没有变化,所以会报错。

如果想要删除元素的话需要使用迭代器内部的 remove 方法:

ArrayList<String> list = new ArrayList<>();
list.add("111");
list.add("222");
list.add("333");
log.info(list.toString());
Iterator<String> it = list.iterator();
while (it.hasNext()){
    String next = it.next();
    //if外使用list的remove方法还是会报错的
    if(next.equals("222")){
        it.remove();//这里使用的是迭代器里面的remove()方法,
        // 当然如果使用list的remove方法在此删除质地感元素的话是成功的,比如:list.remove("222")
    }
}
log.info(list.toString());

结果:

12:06:14.042 [main] INFO com.xiaolinge.com.hello.HelloWord - [111, 222, 333]
12:06:14.046 [main] INFO com.xiaolinge.com.hello.HelloWord - [111, 333]
| 修改元素
使用原始 for:
ArrayList<String> list = new ArrayList<>();
list.add("111");
list.add("222");
list.add("333");
log.info(list.toString());
for (int i = 0; i <list.size(); i++) {
    list.set(i,"444");
}
 log.info(list.toString());

结果:

12:12:56.910 [main] INFO com.xiaolinge.com.hello.HelloWord - [111, 222, 333]
12:12:56.915 [main] INFO com.xiaolinge.com.hello.HelloWord - [444, 444, 444]
哦的 k!可以修改元素。
使用 foreach:
ArrayList<String> list = new ArrayList<>();
 list.add("111");
 list.add("222");
 list.add("333");
 log.info(list.toString());
for (String i : list) {
     i="444";
 }
  log.info(list.toString());

结果:

12:34:47.207 [main] INFO com.xiaolinge.com.hello.HelloWord - [111, 222, 333]
12:34:47.211 [main] INFO com.xiaolinge.com.hello.HelloWord - [111, 222, 333]

看到咯,不行的哦。辣么,修改元素不行,修改元素的属性可不可以呢?让我们来看下吧。

| foreach 修改元素属性(for 就不测试了)

创建一个学生类:

public class Student {
        private int age;
        public int getAge() {
            return age;
        }
        public void setAge(int age) {
            this.age = age;
        }
        public String getName() {
            return name;
        }
        public void setName(String name) {
            this.name = name;
        }
        private String name;
        public Student(){};
        public Student(int age,String name){
            this.age=age;
            this.name=name;
        }
    }

哦的 k,接下来测试代码:

Student student=new Student(1,"huge");
        Student student1=new Student(1,"xiaoyao");
        List<Student> studentList=new ArrayList<Student>();
        studentList.add(student);
        studentList.add(student1);
        System.out.println(student.getName());
        System.out.println(student1.getName());
        for(Student stu:studentList)
        {
            stu.setName("jingtian");
        }
        System.out.println(student.getName());
        System.out.println(student1.getName());

结果:

huge
xiaoyao
jingtian
jingtian

484 很神奇!修改不了对象,却可以修改对象的属性。

总结

如下:

  • for 与 foreach 都可以遍历数组/集合,不过 for 则在较复杂的循环中效率更高。

  • foreach 不可以删除/修改集合元素,而 for 可以。

  • foreach 和 for 都可以修改元素里面的属性。

所以相比较下来 for 循环更为灵活。

-------------  END  -------------

扫码免费获取600+页石杉老师原创精品文章汇总PDF

fcc4dac53d8cdd4dbfa9b9ebca2381c9.png

c3c565534765fd94f1c725b98933ddbc.gif

原创技术文章汇总

15438e587da844b7fed3a3820c85c33c.png

f03c58f7578c5113c72a67d8b7c911a5.png

点个在看你最好看

725dc2c656523549c28c1cea2171aa6b.png

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值