手记

JAVA优化建议(7)

1.List的equals方法只在在乎元素的相不相等

因为equals方法是在AbstractList中,代码如下:

 public boolean equals(Object o) {
        if (o == this)
            return true;
        if (!(o instanceof List))
            return false;

        ListIterator<E> e1 = listIterator();
        ListIterator e2 = ((List) o).listIterator();
        while (e1.hasNext() && e2.hasNext()) {
            E o1 = e1.next();
            Object o2 = e2.next();
            if (!(o1==null ? o2==null : o1.equals(o2)))
                return false;
        }
        return !(e1.hasNext() || e2.hasNext());
    }

因此会得到以下结果

ArrayList arrayList = new ArrayList();
arrayList.add(1);
LinkedList linkedList = new LinkedList();
linkedList.add(1);
System.out.println("ArrayList是否与LinkedList相等"+arrayList.equals(linkedList));

打印结果如下:
ArrayList是否与LinkedList相等true
Map中也一样

2.subList只是返回List的一个视图

源码如下:

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;
        }

        public E set(int index, E e) {
            rangeCheck(index);
            checkForComodification();
            E oldValue = ArrayList.this.elementData(offset + index);
            ArrayList.this.elementData[offset + index] = e;
            return oldValue;
        }

        public E get(int index) {
            rangeCheck(index);
            checkForComodification();
            return ArrayList.this.elementData(offset + index);
        }

        public int size() {
            checkForComodification();
            return this.size;
        }

        public void add(int index, E e) {
            rangeCheckForAdd(index);
            checkForComodification();
            parent.add(parentOffset + index, e);
            this.modCount = parent.modCount;
            this.size++;
        }
        。。。。其他方法省略

subList返回new SubList(),其中add等方法是在原数组上添加
因此会看到如下现象

ArrayList a1 = new ArrayList();
a1.add(1);
List a3 = a1.subList(0,a1.size());
a3.add(2);
System.out.println("a1==a3:"+a1.equals(a3));
for (int i = 0; i <a1.size() ; i++) {
    System.out.println(a1.get(i));
}

打印如下:
a1==a3:true
1 2
但是建立视图以后,不允许修改原List

 ArrayList a1 = new ArrayList();
 a1.add(1);
 List a3 = a1.subList(0,a1.size());
 a3.add(2);
 a1.add(3);

就会报java.util.ConcurrentModificationException
原因如下:

*//ArrayList里面的add方法*
public void add(E e) {
	*//add执行之前会进行并发检查*
    checkForComodification();

    try {
        int i = cursor;
        ArrayList.this.add(i, e);
        cursor = i + 1;
        lastRet = -1;
        expectedModCount = modCount;
    } catch (IndexOutOfBoundsException ex) {
        throw new ConcurrentModificationException();
    }
}

 final void checkForComodification() {
	*//会对比这两个数是否相等*		 
    if (modCount != expectedModCount)
        throw new ConcurrentModificationException();
}

*//但是subList里面的add方法*
 public void add(int index, E e) {
     rangeCheckForAdd(index);
     checkForComodification();
     parent.add(parentOffset + index, e);
     this.modCount = parent.modCount;
     this.size++;
 }
 *没有改expectedModCount,所以会导致校验不通过抛异常*

局部批量删除,可以subList.clear();验证如下

 ArrayList a1 = new ArrayList();
 for (int i = 0; i <100 ; i++) {
     a1.add(Math.random());
 }
 List a3 = a1.subList(0,20);
 a3.clear();
 System.out.println(a1.size());

打印结果如下:
80

0人推荐
随时随地看视频
慕课网APP