ArrayList源码分析

interface Iterable : //实现此接口使集合对象可以通过迭代器遍历自身元素

  Iterator iterator();

public interface Collection extends Iterable: //接口继承接口使用:extends

int size();

boolean isEmpty();

boolean contains(Object o);

boolean containsAll(Collection c);

boolean add(E e);

boolean addAll(Collection c);

boolean remove(Object o);

boolean removeAll(Collection c);

void clear();

boolean equals(Object o);

int hashCode();

boolean retainAll(Collection c);

Iterator iterator();

Object[] toArray();

T[] toArray(T[] a);

public interface List extends Collection

int size();

boolean isEmpty();

boolean contains(Object o);

Iterator iterator();

Object[] toArray();

T[] toArray(T[] a);

boolean add(E e);

boolean remove(Object o);

boolean removeAll(Collection c);

boolean containsAll(Collection c);

boolean addAll(Collection c);

boolean addAll(int index, Collection c);

boolean retainAll(Collection c);

void clear();

boolean equals(Object o);

int hashCode();

E get(int index);

E set(int index, E element);

void add(int index, E element);

E remove(int index);

int indexOf(Object o);

int lastIndexOf(Object o);

ListIterator listIterator();

ListIterator listIterator(int index);

List subList(int fromIndex, int toIndex);

public class ArrayList extends AbstractList

implements List, RandomAccess, Cloneable, java.io.Serializable

private transient Object[] elementData;//实则为数组

        private int size;

public ArrayList() {//默认长度10

this(10);

}

public ArrayList(Collection c) {

elementData = c.toArray(); //现在elementData已经可能不是Object[]

size = elementData.length;

// c.toArray might (incorrectly) not return Object[] (see 6260652)

if (elementData.getClass() != Object[].class)

//将另一个数组copy到当前数组

                    elementData = Arrays.copyOf(elementData, size, Object[].class);

}

public void trimToSize() {

modCount++;

int oldCapacity = elementData.length;//当前长度

                if (size < oldCapacity) {

//当前数组长度大于设定的size,将超过size部分的元素截掉

                    elementData = Arrays.copyOf(elementData, size);

}

}

//数组扩容/收缩[自我拷贝]

public static T[] copyOf(T[] original, int newLength) {

return (T[]) copyOf(original, newLength, original.getClass());

}

public boolean isEmpty() {

return size == 0;//是否为空,即大小是否为0

}

public boolean contains(Object o) {

return indexOf(o) >= 0;//contains()底层调用的是indexOf()

}

public Object clone() {

try {

@SuppressWarnings("unchecked")

//浅克隆:只复制了基本类型字段

                        ArrayList v = (ArrayList) super.clone();//???为什么克隆之后还要将内容拷贝过去

                        //深克隆:引用类型字段也复制

                        v.elementData = Arrays.copyOf(elementData, size);

v.modCount = 0;

return v;

} catch (CloneNotSupportedException e) {

// this shouldn't happen, since we are Cloneable

throw new InternalError();

}

}

public Object[] toArray() {

return Arrays.copyOf(elementData, size);

}

public T[] toArray(T[] a) {

if (a.length < size) //小于elementData长度时,通过Arrays.copyOf()复制出一个新的数组

                        return (T[]) Arrays.copyOf(elementData, size, a.getClass());

System.arraycopy(elementData, 0, a, 0, size);

if (a.length > size)

a[size] = null;

return a;

}

//private:同类, default:同包, protected:不同包子类, public:公共

            E elementData(int index) {

return (E) elementData[index];

}

public E get(int index) {

rangeCheck(index); //throw new IndexOutOfBoundsException(outOfBoundsMsg(index));

return elementData(index);

}

public E set(int index, E element) {

rangeCheck(index);

E oldValue = elementData(index);

elementData[index] = element;

return oldValue;

}

public boolean add(E e) {

ensureCapacityInternal(size + 1);  // Increments modCount!!

elementData[size++] = e;

return true;

}

private void ensureCapacityInternal(int minCapacity) {

modCount++;

// overflow-conscious code

if (minCapacity - elementData.length > 0)//elementData.length:这个不是数组实际长度??

                        grow(minCapacity);//超出容量,扩容

                }

//防溢出

            private void grow(int minCapacity) {

// overflow-conscious code

int oldCapacity = elementData.length;

// >>位运算,右移动一位。 整体相当于newCapacity =oldCapacity + 0.5 * oldCapacity

// jdk1.7采用位运算比以前的计算方式更快

                    int newCapacity = oldCapacity + (oldCapacity >> 1);

if (newCapacity - minCapacity < 0) //这里为什么不直接用<【minCapacity可能为负数】

                        newCapacity = minCapacity;

if (newCapacity - MAX_ARRAY_SIZE > 0) //这里为什么不直接用>

newCapacity = hugeCapacity(minCapacity);

// minCapacity is usually close to size, so this is a win:

elementData = Arrays.copyOf(elementData, newCapacity);//自我拷贝

                }

public void add(int index, E element) {

rangeCheckForAdd(index);

ensureCapacityInternal(size + 1);  // Increments modCount!!

//将index之后的元素,拷到index+1的位置之后

                    System.arraycopy(elementData, index, elementData, index + 1,

size - index);

//空出index的位置,将新元素加进来

                    elementData[index] = element;

size++;

}

public E remove(int index) {

rangeCheck(index);

modCount++;

E oldValue = elementData(index);

int numMoved = size - index - 1;

if (numMoved > 0)

System.arraycopy(elementData, index+1, elementData, index,

numMoved);

elementData[--size] = null; // Let gc do its work

return oldValue;

}

public boolean addAll(Collection c) {

Object[] a = c.toArray();

int numNew = a.length;

ensureCapacityInternal(size + numNew);  // Increments modCount

System.arraycopy(a, 0, elementData, size, numNew);//将新加的即可拷到原数组后面

                    size += numNew;

return numNew != 0;

}

public boolean addAll(int index, Collection c) {

rangeCheckForAdd(index);

Object[] a = c.toArray();

int numNew = a.length;

ensureCapacityInternal(size + numNew);  // Increments modCount

int numMoved = size - index;

if (numMoved > 0)

System.arraycopy(elementData, index, elementData, index + numNew,

numMoved);

System.arraycopy(a, 0, elementData, index, numNew);

size += numNew;

return numNew != 0;

}

protected void removeRange(int fromIndex, int toIndex) {

modCount++;

int numMoved = size - toIndex;

System.arraycopy(elementData, toIndex, elementData, fromIndex,

numMoved);

// Let gc do its work

int newSize = size - (toIndex-fromIndex);

while (size != newSize)

elementData[--size] = null;

}

private class Itr implements Iterator {

int cursor;      // index of next element to return

int lastRet = -1; // index of last element returned; -1 if no such

//将arrayList被修改的次数记录下来,如果Itr遍历过程中,modCount被修改,直接抛出异常

                        int expectedModCount = modCount;

public void remove() {

if (lastRet < 0)

throw new IllegalStateException();

checkForComodification();

try {

/*静态内部类的话,只能new一个外部类对象了。

若是实例内部类,可以使用类名限定

例子:

                              (外部类类名)OuterClass.this就可以得到外部类的引用了。*/

ArrayList.this.remove(lastRet);

cursor = lastRet;

lastRet = -1;

//迭代器能过正常删除元素的原因,重置expectedModCount

expectedModCount = modCount;

} catch (IndexOutOfBoundsException ex) {

throw new ConcurrentModificationException();

}

}

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 218,640评论 6 507
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 93,254评论 3 395
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 165,011评论 0 355
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,755评论 1 294
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,774评论 6 392
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,610评论 1 305
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,352评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,257评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,717评论 1 315
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,894评论 3 336
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 40,021评论 1 350
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,735评论 5 346
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,354评论 3 330
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,936评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 33,054评论 1 270
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,224评论 3 371
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,974评论 2 355

推荐阅读更多精彩内容