java_集合体系之List体系总结、应用场景――07(三)

2014-11-24 07:37:06 · 作者: · 浏览: 1
dex处的插入、删除、ArrayList和LinkedList都是先通过索引查找到指定位置、然后进行下一步的插入删除操作、上面我们知道LinkedList是先通过二分法查找index范围再确定index具体位置、但是ArrayList是直接定位到index处、为什么LinkedList反而快?依然通过源码找原因。

ArrayList关于指定位置的元素的插入:

    /**
     * 确保此ArrayList的最小容量能容纳下参数minCapacity指定的容量、
     * 1、minCapacity大于原来容量、则将原来的容量增加(oldCapacity * 3)/2 + 1;
     * 2、若minCapacity仍然大于增加后的容量、则使用minCapacity作为ArrayList容量
     * 3、若minCapacity不大于增加后的容量、则使用增加后的容量。
     */
    public void ensureCapacity(int minCapacity) {
		modCount++;
		int oldCapacity = elementData.length;
		if (minCapacity > oldCapacity) {
		    Object oldData[] = elementData;
		    int newCapacity = (oldCapacity * 3)/2 + 1;
	    	    if (newCapacity < minCapacity)
	    	    	newCapacity = minCapacity;
	            // minCapacity is usually close to size, so this is a win:
	            elementData = Arrays.copyOf(elementData, newCapacity);
		}
    }
    /** 将指定元素添加到指定的索引处 、
     *	注意:
     *	1、如果指定的index大于Object[] 的size或者小于0、则抛IndexOutOfBoundException
     *	2、检测Object[]是否需要扩容
     *	3、 将从index开始到最后的元素后移一个位置、
     *	4、将新添加的元素添加到index去。
     */
    public void add(int index, E element) {
		if (index > size || index < 0)
		    throw new IndexOutOfBoundsException(
			"Index: "+index+", Size: "+size);
	
		ensureCapacity(size+1);  // Increments modCount!!
		System.arraycopy(elementData, index, elementData, index + 1,
				 size - index);
		elementData[index] = element;
		size++;
    }

LinkedList关于指定位置的元素的插入:

    /** 在index前添加节点,且节点的值为element*/
    public void add(int index, E element) {
        addBefore(element, (index==size   header : entry(index)));
    }
    /**  获取双向链表LinkedList中指定位置的节点、是LinkedList实现List中通过index操作元素的关键*/
    private Entry
  
    entry(int index) {
        if (index < 0 || index >= size)
            throw new IndexOutOfBoundsException("Index: "+index+ ", Size: "+size);
        Entry
   
     e = header; if (index < (size >> 1)) { for (int i = 0; i <= index; i++) e = e.next; } else { for (int i = size; i > index; i--) e = e.previous; } return e; } //新建节点、节点值是e、将新建的节点添加到entry之前 private Entry
    
      addBefore(E e, Entry
     
       entry) { //觉得难理解的可以先花个几分钟看一下链式结构资料、最好是图片形式的 //新建节点实体 Entry
      
        newEntry = new Entry
       
        (e, entry, entry.previous); //将参照节点原来的上一个节点(即插在谁前面的)的下一个节点设置成newEntry newEntry.previous.next = newEntry; //将参照节点(即插在谁前面的)的前一个节点设置成newEntry newEntry.next.previous = newEntry; size++; modCount++; return newEntry; }
       
      
     
    
   
  

对比上面代码可以看出来ArrayList每当插入一个元素时、都会调用System.arraycopy()将指定位置后面的所有元素后移一位、重新构造一个数组、这是比较消耗资源的、而LinkedList是直接改变index前后元素的上一个节点和下一个节点的引用、而不需要动其他的东西、所以效率很高。

ArrayList与Vector:

ArrayList、Vector都是继承与AbstractList、并且在类结构上没有多少差异、但是因为Vector要同步方法、所以在性能上不如ArrayList、从源码也可以看出Vector许多方法都是使用关键字synchronized修饰的。不再贴源码

总结:


学以致用、最后总结下上述List集合体系的各个类的使用环境:

1、当需要对集合进行大量的查询时、并且是单线程环境下使用ArrayList

2、当需要对集合进行大量添加、删除时、并且是单线程环境下使用LinkedList、

3、当多线程时、需要对集合进行大量的查询时、可以考虑使用Vector或者Stack、但是不建议、我们可以使用多次提到的Collections类包装。


更多内容:java_集合体系之总体目录――00