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

简介:

List接口提供了subList方法,其作用是返回一个列表的子列表.这与String类的subString有点类似.但是他们的功能是否相同?看代码:

复制代码
 1 import java.util.ArrayList;
 2 import java.util.List;
 3 
 4 public class Client {
 5     public static void main(String[] args) {
 6         //定义一个包含两个字符串的列表
 7         List<String> c = new ArrayList<String>();
 8         c.add("A");
 9         c.add("B");    
10         //构造一个包含c的字符串列表
11         List<String> c1 = new ArrayList<String>(c);
12         //通过subList生成与c相同的列表
13         List<String> c2 = c.subList(0, c.size());
14         //c2增加一个元素
15         c2.add("C");        
16         System.out.println("c == c1? " + c.equals(c1));
17         System.out.println("c == c2? " + c.equals(c2));
18         
19     }
20 }
复制代码

程序输出:

c == c1? false
c == c2? true

c1是通过ArrayList的构造函数创建的,c2是通过列表的subList方法创建的,然后c2又增加了一个元素C,现在的问题是输出的结果是什么呢?

列表c与c1,c2之间是什么关系呢?

别忙着回答这个问题,回想一下String类的subString方法,看看它是如何工作的.

复制代码
1 public class Client {
2     public static void main(String[] args) {
3         String str = "AB";
4         String str1 = new String(str);
5         String str2 = str.substring(0) + "C";
6         System.out.println("str == str1 " + str.equals(str1));
7         System.out.println("str == str2 " + str.equals(str2));
8     }
9 }
复制代码

输出结果:

str == str1 true
str == str2 false

很明显str与str1是相等的(虽然不是同一个对象,但用equals方法判断是相等的),但它们与str2不相等,这也毋庸置疑,因为str2在对象池中重新生成了一个新的对象,其值是ABC,那当然与str和str1不相等了.

为什么subList的结果正好好subString的相反?

c2是通过subList方法从c列表中生成了 一个子列表,然后c2又增加了一个元素,可为什么增加了 一个元素还会相等呢?看subList的源码:

复制代码
1     public List<E> subList(int fromIndex, int toIndex) {
2         return (this instanceof RandomAccess ?
3                 new RandomAccessSubList<>(this, fromIndex, toIndex) :
4                 new SubList<>(this, fromIndex, toIndex));
5     }
复制代码

subList方法由AbstractList实现的,它会根据是不是可以随机存取来提供不同的SubList实现方式,不过,随机存储的使用频率比较高,而且RandomAccessSubList也是SubList子类,所以所有的操作都是由SubList类实现的(除了自身的SubList方法外),

看SubList的源码:

 

通过阅读这段代码,就非常清楚的subList方法的实现原理了:它返回的SubList类也是AbstractList的子类,其所有的方法如get,set,add,remove等都是在原始列表上的操作,它自身并没有生成一个数组或是链表,也就是子列表只是原始列表的一个视图(View),所有的修改动作都反映在了原列表上....

我们例子中的c2增加了一个元素C,不过增加的元素C到了c列表上,两个变量的元素仍然保持完全一致,相等也就很自然了.

 

为什么c和c1不相等.因为通过ArrayList构造函数创建的List对象c1实际上是新列表,它是通过数组的copyOf动作生成的,所生成的列表c1与原列表c之间没有任何关系(虽然是浅拷贝,但元素类型是String,也就是说元素是深拷贝的),

然后c又增加了元素,因为c1与c之间已经没有任何的关系了,自然不相等.

 

subList产生的列表只是一个视图,所有的修改动作直接作用于原列表.

 


本文转自SummerChill博客园博客,原文链接:http://www.cnblogs.com/DreamDrive/p/5660044.html,如需转载请自行联系原作者

相关文章
|
6月前
|
小程序 JavaScript
小程序循环列表删除当前选中列表的方法
小程序循环列表删除当前选中列表的方法
78 0
|
11月前
|
数据库
因祸得福——创建视图时改变字段的类型
因祸得福——创建视图时改变字段的类型
106 0
|
12月前
|
Python
python实现将给定列表划分为元素和大致相等的两个子列表
python实现将给定列表划分为元素和大致相等的两个子列表
|
C++
新建项无资源字典类型
新建项无资源字典类型
96 0
新建项无资源字典类型
|
数据库 开发者 索引
列表增加元素|学习笔记
快速学习列表增加元素
104 0
列表增加元素|学习笔记
评论列表案例-将评论Item项抽离为单独的CmtItem组件
评论列表案例-将评论Item项抽离为单独的CmtItem组件
SwiftUI—如何通过AnyView返回任意类型的视图
SwiftUI—如何通过AnyView返回任意类型的视图
206 0
SwiftUI—如何通过AnyView返回任意类型的视图
7.5.1 创建条目列表示例
7.5.1 创建条目列表示例
78 0
7.5.1 创建条目列表示例
|
JavaScript 前端开发
[译] 从 0 创建自定义元素
在上一篇文章,我们在文档中创建了 HTML 模板,希望它们在需要时才呈现,这让我们开始接触 Web 组件。 接下来,我们将继续创建对话框组件的自定义元素版本,该自定义元素版本目前仅使用 HTMLTemplateElement。
628 0
|
C#
绑定到列表的指定元素
原文:绑定到列表的指定元素                                绑定到列表的指定元素                                   周银辉 在WPF中,假设有一个对象People,我们要将某元素的某个属性绑定到它的Name属性的话,Path就是"Name”,但如果我们要绑定要List的第i个元素呢,应该这样写:".
556 0