解析java.util集合类源码(SubList)

本文深入探讨了Java中的SubList类,它是List接口的一个实现,用于创建原List对象的视图。SubList与原List之间的同步问题在文章中得到详细解释,包括其属性、构造方法以及在修改时如何引发`ConcurrentModificationException`。通过实例代码和调试过程,帮助读者更好地理解SubList的工作原理。
摘要由CSDN通过智能技术生成

SubList类


在1.6和1.7的api中没有给出SubList的说明,只有subList方法,返回一个List列表的子类列表

public List<E> subList(int fromIndex,int toIndex)
但在jdk的源码中,你可以搜到SubList,它在AbstractList类下,却不是内部类

先看下SubList的结构

SubList继承AbstractList抽象类,AbstractList实现了List接口,所以SubList说到底就是一个List的实现类,主要用于返回List的视图,这个视图是原List对象中的一部分,确实是一部分,直接将原List对象引用到新的子视图的List,对子视图进行改变,原List对象也会随之改变

下面就解释为什么会造成这种情况

首先看SubList中的属性域

SubList有4个自己的属相域和1个继承自AbstractList的modCount

private AbstractList<E> l;
private int offset;
private int size;
private int expectedModCount;
AbstractList l 就是存放母List的那个引用,当List对象(本编中指类似ArrayList的实现List的对象)调用subList方法时候,会调用abstractList中的subList的实现

 //AbstractList中的subList方法
 public List<E> subList(int fromIndex, int toIndex) {
        return (this instanceof RandomAccess ?
                new RandomAccessSubList<E>(this, fromIndex, toIndex) :
                new SubList<E>(this, fromIndex, toIndex));
    }
这里this是多态的,指的是那个外部调用subList方法的那个List对象,先判断this是否实现了RandomAccess接口,实现返回RandomAccessSublist否则返回SubList,通过构造方法将this传给SubList中l属性,在介绍构造方法时细说

int offset 开始子List在母List中的开始位置

int size 记录大小 = toIndex - fromIndex

int expectedModCount  列表的中元素添加、删除的次数,子列表的同步使用

 SubList(AbstractList<E> list, int fromIndex, int toIndex) {
        if (fromIndex < 0)
            throw new IndexOutOfBoundsException("fromIndex = " + fromIndex);
        if (toIndex > list.size())
            throw new IndexOutOfBo
  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
这个问题是因为 `subList` 方法返回的是 `ArrayList$SubList` 类型,而不是 `ArrayList` 类型。你可以通过将它转换为 `ArrayList` 类型来解决这个问题。 下面是一个示例代码,展示如何将 `subList` 返回的 `ArrayList$SubList` 转换为 `ArrayList`: ```java import java.util.ArrayList; import java.util.Collections; import java.util.List; public class Main { public static void main(String[] args) { // 假设这是原始的脉搏波数据 ArrayList<Double> pulseData = new ArrayList<>(); pulseData.add(10.0); pulseData.add(20.0); pulseData.add(15.0); pulseData.add(25.0); pulseData.add(12.0); // 对脉搏波数据进行中值滤波 ArrayList<Double> filteredData = medianFilter(pulseData); // 打印滤波后的数据 for (Double data : filteredData) { System.out.println(data); } } public static ArrayList<Double> medianFilter(ArrayList<Double> data) { ArrayList<Double> filteredData = new ArrayList<>(); int windowSize = 3; // 滑动窗口大小 for (int i = 0; i < data.size(); i++) { int startIndex = Math.max(0, i - windowSize / 2); int endIndex = Math.min(data.size() - 1, i + windowSize / 2); // 获取滑动窗口内的数据,并转换为 ArrayList 类型 List<Double> window = new ArrayList<>(data.subList(startIndex, endIndex + 1)); // 对滑动窗口内的数据进行排序 Collections.sort(window); // 取中值作为滤波后的数据 double median = window.get(window.size() / 2); filteredData.add(median); } return filteredData; } } ``` 在上述代码中,我们将 `subList` 返回的结果使用 `ArrayList` 的构造函数转换为 `ArrayList` 类型,然后进行后续的处理。 请尝试使用这个方法,看看是否能够解决你的问题。如果你还有其他问题,请随时追问。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值