ArrayList的sublist();方法的陷阱
2016-04-17 19:19
274 查看
java.util.List中有一个subList方法,用来返回一个list的一部分的视图。
它返回原来list的从[fromIndex, toIndex)之间这一部分的List(下面称之为sublist),但是这个sublist是依赖于原来的List集合
所以,你对原来的list和返回的sublist做的“非结构性修改”(non-structural changes),都会影响到彼此对方。
所谓的“非结构性修改”,是指不涉及到list的大小改变的修改。相反,结构性修改,指改变了list大小的修改。
看看ArrayList 的源码:
那么,如果涉及到结构性修改会怎么样呢?
如果发生结构性修改的是返回的子list,那么原来的list的大小也会发生变化;
而如果发生结构性修改的是原来的list(不包括由于返回的子list导致的改变),那么返回的子list语义上将会是undefined。在AbstractList(ArrayList的父类)中,undefined的具体表现形式是抛出一个ConcurrentModificationException。
因此,如果你在调用了sublist返回了子list之后,如果修改了原list的大小,那么之前产生的子list将会失效,变得不可使用。
tips: 如何删除一个list的某个区段,比如删除list的第2-5个元素?
方法是: 可以利用sublist的幕后还是原来的list的这个特性,比如
这样就可以了。
参考:
http://www.cnblogs.com/gaojing/archive/2012/06/17/java-list-sublist-caution.html
一个很有趣的思考:如何最高效的实现一个list的split方法?
参见:http://stackoverflow.com/questions/379551/java-split-a-list-into-two-sub-lists。
List<E> subList(int fromIndex, int toIndex);
它返回原来list的从[fromIndex, toIndex)之间这一部分的List(下面称之为sublist),但是这个sublist是依赖于原来的List集合
所以,你对原来的list和返回的sublist做的“非结构性修改”(non-structural changes),都会影响到彼此对方。
所谓的“非结构性修改”,是指不涉及到list的大小改变的修改。相反,结构性修改,指改变了list大小的修改。
看看ArrayList 的源码:
1: private class SubList 是内部类,subList返回的就是这个类的实例 2:SubList内部类的构造方法的参数parent就是subList方法中的第二行传入的this 3:因此Sublist内部类和ArrayList的内部保存数据的地址是一样得;即它们在内存中是同一个List(集合),只是parentOffset ,size等参数不同
public List<E> subList(int fromIndex, int toIndex) { subListRangeCheck(fromIndex, toIndex, size); return new SubList(this, 0, fromIndex, toIndex); } private class SubList extends AbstractList<E> implements RandomAccess { private final AbstractList<E> parent; private final int parentOffset; private final int offset; int size; SubList(AbstractList<E> parent, int offset, int fromIndex, int toIndex) { this.parent = parent; this.parentOffset = fromIndex; this.offset = offset + fromIndex; this.size = toIndex - fromIndex; this.modCount = ArrayList.this.modCount; } //省略 }
那么,如果涉及到结构性修改会怎么样呢?
如果发生结构性修改的是返回的子list,那么原来的list的大小也会发生变化;
而如果发生结构性修改的是原来的list(不包括由于返回的子list导致的改变),那么返回的子list语义上将会是undefined。在AbstractList(ArrayList的父类)中,undefined的具体表现形式是抛出一个ConcurrentModificationException。
因此,如果你在调用了sublist返回了子list之后,如果修改了原list的大小,那么之前产生的子list将会失效,变得不可使用。
tips: 如何删除一个list的某个区段,比如删除list的第2-5个元素?
方法是: 可以利用sublist的幕后还是原来的list的这个特性,比如
list.subList(from, to).clear();
这样就可以了。
参考:
http://www.cnblogs.com/gaojing/archive/2012/06/17/java-list-sublist-caution.html
一个很有趣的思考:如何最高效的实现一个list的split方法?
参见:http://stackoverflow.com/questions/379551/java-split-a-list-into-two-sub-lists。
相关文章推荐
- Python语句和控制流
- Cyclic Nacklace(HDU 3746)
- hbase安装过程详解
- Oracle数据库实例的启动及关闭
- 设计模式与实例代码:Proxy模式
- 推荐一个站点:里面有很多不错的长文
- 我所理解的单例模式
- 不平衡的振动诊断要点
- 新视图RecyclerView(一)
- for循环问题
- #OC---一些基本概念
- H.264码流结构解析
- Python 爬虫学习2
- 浅谈单调队列优化dp
- 第二会
- PIE结对编程
- linux find 几个特殊点
- 使用 matlab 数字图像处理(九)—— 去卷积(deconvolution,逆滤波复原)
- python文件读写
- Array,Collection,Map的浅析