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