其他分享
首页 > 其他分享> > 注意!在subList生成子列表之后,一定不要随便更改原列表

注意!在subList生成子列表之后,一定不要随便更改原列表

作者:互联网

watermark,size_16,text_QDUxQ1RP5Y2a5a6i,color_FFFFFF,t_100,g_se,x_10,y_10,shadow_90,type_ZmFuZ3poZW5naGVpdGk=

watermark,size_16,text_QDUxQ1RP5Y2a5a6i,color_FFFFFF,t_100,g_se,x_10,y_10,shadow_90,type_ZmFuZ3poZW5naGVpdGk=

大家好,我是雄雄。

前几期我们说过,subList方法是返回原列表的子列表,并且我们还说过,在subList返回的子列表上操作时,会直接影响着原列表,原文在这里:

subList?? subString???

子列表只是原列表的一个视图

那么,大家有没有想过这样一个问题,在调用subList方法返回子列表之后,我们要是不操作子列表,而是操作的原列表,会怎么样呢?

下面我们先从一段代码上看起:

public static void testSubList(){
    //初始化一个集合
    List<String> lists = new ArrayList<String>();
    //给集合中添加四个元素
    lists.add("A");
    lists.add("B");
    lists.add("C");
    lists.add("D");
    //遍历集合
    System.out.println("原来集合中的元素:");
    for (String str: lists) {
      System.out.print(str+" ");
    }
    System.out.println("\n子列表中的元素:");
    List<String>new_list = lists.subList(0, 2);
    
    for (String str: new_list) {
      System.out.print(str+" ");
    }
  }

运行结果如下:

watermark,size_16,text_QDUxQ1RP5Y2a5a6i,color_FFFFFF,t_100,g_se,x_10,y_10,shadow_90,type_ZmFuZ3poZW5naGVpdGk=

这块代码很好理解,无外乎就是初始化了个集合,并添加了几个元素,通过subList方法返回了一个子列表,最后将两个列表中的元素都遍历出来。

接下来我要改一下代码了,如下:

public static void testSubList(){
    //初始化一个集合
    List<String> lists = new ArrayList<String>();
    //给集合中添加四个元素
    lists.add("A");
    lists.add("B");
    lists.add("C");
    lists.add("D");
    //遍历集合
    System.out.println("原来集合中的元素:");
    for (String str: lists) {
      System.out.print(str+" ");
    }
    System.out.println("\n子列表中的元素:");
    List<String>new_list = lists.subList(0, 2);
    
    for (String str: new_list) {
      System.out.print(str+" ");
    }
    //给原来的集合中新加一个元素
    lists.add("E");
    System.out.println("\n新加元素之后原来集合的元素:");
    for (String str: lists) {
      System.out.print(str+" ");
    }
    
    System.out.println("\n新加元素之后子列表中的元素:");
    for (String str: new_list) {
      System.out.print(str+" ");
    }
  }

这段代码前面部分没有变,后面只是给原集合中新插入了一个元素“E”,然后对原集合以及子列表进行遍历,运行结果如下所示:

watermark,size_16,text_QDUxQ1RP5Y2a5a6i,color_FFFFFF,t_100,g_se,x_10,y_10,shadow_90,type_ZmFuZ3poZW5naGVpdGk=

居然报错了?并且是在原集合中新加元素之后遍历子列表时报的错。ConcurrentModificationException是并发修改异常,但是我们这里并没有多线程操作,何来并发异常?其实,原因很简单,那就是subList方法是原列表的子列表,当原来的集合(原列表)修改之后,subList取出的子列表并未跟着一起修改,也就是不会生成新列表,最后在最字列表操作时,程序就会发现修改计数器(Modification)与预期的不符合,故抛出此异常。

因此,在subList生成子列表之后,一定不要随便更改原列表。

往期精彩

神奇!一行代码实现删除某集合下标20-30的元素

2020-10-14

watermark,size_16,text_QDUxQ1RP5Y2a5a6i,color_FFFFFF,t_100,g_se,x_10,y_10,shadow_90,type_ZmFuZ3poZW5naGVpdGk=

‘小会计’的转行之旅

2020-10-13

watermark,size_16,text_QDUxQ1RP5Y2a5a6i,color_FFFFFF,t_100,g_se,x_10,y_10,shadow_90,type_ZmFuZ3poZW5naGVpdGk=

班级日常分享,一天一瞬间

2020-10-14

watermark,size_16,text_QDUxQ1RP5Y2a5a6i,color_FFFFFF,t_100,g_se,x_10,y_10,shadow_90,type_ZmFuZ3poZW5naGVpdGk=

子列表只是原列表的一个视图

2020-10-12

watermark,size_16,text_QDUxQ1RP5Y2a5a6i,color_FFFFFF,t_100,g_se,x_10,y_10,shadow_90,type_ZmFuZ3poZW5naGVpdGk=

“半路出家”的程序猿怎么不被“熊”

2020-10-11

watermark,size_16,text_QDUxQ1RP5Y2a5a6i,color_FFFFFF,t_100,g_se,x_10,y_10,shadow_90,type_ZmFuZ3poZW5naGVpdGk=

某同学工作之后的感悟

2020-10-10

watermark,size_16,text_QDUxQ1RP5Y2a5a6i,color_FFFFFF,t_100,g_se,x_10,y_10,shadow_90,type_ZmFuZ3poZW5naGVpdGk=

班级日常分享,一天一瞬间

2020-10-10

watermark,size_16,text_QDUxQ1RP5Y2a5a6i,color_FFFFFF,t_100,g_se,x_10,y_10,shadow_90,type_ZmFuZ3poZW5naGVpdGk=

watermark,size_16,text_QDUxQ1RP5Y2a5a6i,color_FFFFFF,t_100,g_se,x_10,y_10,shadow_90,type_ZmFuZ3poZW5naGVpdGk=

watermark,size_16,text_QDUxQ1RP5Y2a5a6i,color_FFFFFF,t_100,g_se,x_10,y_10,shadow_90,type_ZmFuZ3poZW5naGVpdGk=

点分享

watermark,size_16,text_QDUxQ1RP5Y2a5a6i,color_FFFFFF,t_100,g_se,x_10,y_10,shadow_90,type_ZmFuZ3poZW5naGVpdGk=

点点赞

watermark,size_16,text_QDUxQ1RP5Y2a5a6i,color_FFFFFF,t_100,g_se,x_10,y_10,shadow_90,type_ZmFuZ3poZW5naGVpdGk=

点在看

标签:更改,System,lists,subList,str,列表,out
来源: https://blog.51cto.com/u_12718584/2865760