分享
 
 
 

ArrayList源码分析

王朝java/jsp·作者佚名  2006-01-09
窄屏简体版  字體: |||超大  

终于可以开始分析第一个具体的类,我们对ArrayList应该非常面熟了,不管你是否了解它是如何实现的,

但是我们到处都使用到它。

声明如下:

public class ArrayList extends AbstractListimplements List, RandomAccess, Cloneable, java.io.Serializable

有关AbstractList:http://blog.csdn.net/treeroot/archive/2004/09/14/104743.aspx

有关List: http://blog.csdn.net/treeroot/archive/2004/09/14/104638.aspx

有关RandomAccess:http://blog.csdn.net/treeroot/archive/2004/09/14/104538.aspx

有关Cloneable :http://blog.csdn.net/treeroot/archive/2004/09/07/96936.aspx

有关java.io.Serializeable:主要用于对象序列化。

private static final long serialVersionUID = 8683452581122892189L;

版本控制

private transient Object elementData[];

内部结构,原来就是一个数组,这里不让它串行化。

private int size;

该列表的大小,也就是元素个数。

public ArrayList(int initialCapacity) {

super();

if (initialCapacity < 0)

throw new IllegalArgumentException("Illegal Capacity: "+initialCapacity);

this.elementData = new Object[initialCapacity];

}

构造函数,初始化内部数组为指定大小,注意列表是空的。

public ArrayList() {

this(10);

}

默认初始话内部数组大小为10,为什么是10是没有理由的,可能比较合适吧。

public ArrayList(Collection c) {

size = c.size();

// Allow 10% room for growth

elementData = new Object[(int)Math.min((size*110L)/100,Integer.MAX_VALUE)];

c.toArray(elementData);

}

通过一个集合来初始话该ArrayList,内部数组申请的空间比c大,主要是为了提高效率。注意到c.toArray(elementData)

方法的调用,这里肯定不会生成新的数组,如果用elementData=c.toArray()效率就差不少了。另外这里调用了Math的静态

方法min来获得较小值。

public void trimToSize() {

modCount++;

int oldCapacity = elementData.length;

if (size < oldCapacity) {

Object oldData[] = elementData;

elementData = new Object[size];

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

}

}

这个方法去掉多余的空间,使内部数组的大小刚好等于ArrayList的size(),这个方法需要重新分配空间,而已需要一个数组

拷贝过程(arraycopy是一个native方法,用的比较多),一般情况下这个方法很少被调用。

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;

elementData = new Object[newCapacity];

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

}

}

这个方法来扩大ArrayList的容量,使它至少能容纳minCapacity个元素,如果数组容量大于该值,什么也不做。否则按某个

算法(1.5倍加1)增加,如果不够minCapacity大的话,就设置为minCapacity。这个方法在add和addAll方法中都要调用,

这里为什么设置为public呢?因为每次重新分配空间都是比较消耗时间的(new操作还要arrayCopy),如果能预计可能的大小

的话,这个方法就有比较的灵活性。虽然该扩容算发已经比较好,但是还是可以通过自己的控制提高效率,这个方法为程序员

带来的方便。

eg1:

ArrayList al=new ArrayList();

for(int i=0;i<100;i++){

Object obj=new Object();

al.add(obj);

}

eg2(更高效):

ArrayList al=new ArrayList(100);

for(int i=0;i<100;i++){

Object obj=new Object();

al.add(obj);

}

或者

ArrayList al=new ArrayList();

al.ensureCapacity(100);

for(int i=0;i<100;i++){

Object obj=new Object();

al.add(obj);

}

public int size() {

return size;

}

返回大小

public boolean isEmpty() {

return size == 0;

}

是否为空

public boolean contains(Object elem) {

return indexOf(elem) >= 0;

}

是否包含指定元素,调用的是indexOf()方法。

public int indexOf(Object elem) {

if (elem == null) {

for (int i = 0; i < size; i++)

if (elementData[i]==null)

return i;

} else {

for (int i = 0; i < size; i++)

if (elem.equals(elementData[i]))

return i;

}

return -1;

}

这个方法遍历列表(数组0..size-1)

public int lastIndexOf(Object elem) {

if (elem == null) {

for (int i = size-1; i >= 0; i--)

if (elementData[i]==null)

return i;

} else {

for (int i = size-1; i >= 0; i--)

if (elem.equals(elementData[i]))

return i;

}

return -1;

}

这个方法和上面的基本一样,顺序不一样而已。

public Object clone() {

try {

ArrayList v = (ArrayList)super.clone();

v.elementData = new Object[size];

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

v.modCount = 0;

return v;

} catch (CloneNotSupportedException e) {

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

throw new InternalError();

}

}

覆盖Object中的clone()方法,实现clone,注意这里是一个浅拷贝,两个ArrayList中的数组中的元素

是相同的,因为System.arraycopy就是浅拷贝。

public Object[] toArray() {

Object[] result = new Object[size];

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

return result;

}

返回ArrayList元素的一个数组,注意这里虽然生成了一个新的数组,但是数组元素和集合中的元素是共享的,

Collection接口中说这个是安全的是不严格的,下面的例子演示了这个效果。

eg1:

ArrayList al=new ArrayList();

al.add(new StringBuffer("hello"));

Object[] a=al.toArray();

StringBuffer sb=(StringBuffer)a[0];

sb.append("changed"); //改变数组元素同样也改变了原来的ArrayList中的元素

System.out.println(al.get(0));

这里不要用String来代替StringBuffer,因为String是常量。

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

if (a.length < size)

a = (Object[])java.lang.reflect.Array.newInstance(a.getClass().getComponentType(), size);

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

if (a.length > size)

a[size] = null;

return a;

}

这个方法有可能不需要生成新的数组,注意到如果数组a容量过大,只在size处设置为null。

public Object get(int index) {

RangeCheck(index);

return elementData[index];

}

可以看随机访问效率是很高的,和数组的索引访问是一样的,方式设计到索引值都会先检查。

public Object set(int index, Object element) {

RangeCheck(index);

Object oldValue = elementData[index];

elementData[index] = element;

return oldValue;

}

更新指定位置的值,并访问原来的值。

public boolean add(Object o) {

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

elementData[size++] = o;

return true;

}

添加一个新的元素到末尾,前面说道新增方法都要先调用ensureCapacity方法,这里没有调用add(size,o)方法。

public void add(int index, Object 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++;

}

在指定位置插入元素,指定元素和后面的元素后移。

public Object remove(int index) {

RangeCheck(index);

modCount++;

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

}

删除指定位置的元素,后面的元素前移,返回被删除的元素,这里要注意的elementData[--size]=null这条语句,

如果不这样的话,就有可能造成内存泄露,因为该对象其实已经没有用,但是内部还有一个它的引用,如果不设置

为null,GC就无法回收这个空间,积累多了就有可能造成内存泄露,这里只说有可能,而不是一定。

public void clear() {

modCount++;

// Let gc do its work

for (int i = 0; i < size; i++)

elementData[i] = null;

size = 0;

}

这段代码比较高效,这里也必须设置为空引用,理由同上。

public boolean addAll(Collection c) {

modCount++;

int numNew = c.size();

ensureCapacity(size + numNew);

Iterator e = c.iterator();

for (int i=0; i

elementData[size++] = e.next();

return numNew != 0;

}

添加集合c中的元素到ArrayList的末尾,添加成功返回true,如果集合c为空,返回false。

public boolean addAll(int index, Collection c) {

if (index > size || index < 0)

throw new IndexOutOfBoundsException("Index: "+index+", Size: "+size);

int numNew = c.size();

ensureCapacity(size + numNew); // Increments modCount!!

int numMoved = size - index;

if (numMoved > 0)

System.arraycopy(elementData, index, elementData, index + numNew,numMoved);

Iterator e = c.iterator();

for (int i=0; i

elementData[index++] = e.next();

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;

}

这是一个保护方法,删除指定位置fromIndex到toIndex的元素,包括前面不包括后面。

private void RangeCheck(int index) {

if (index >= size || index < 0)

throw new IndexOutOfBoundsException("Index: "+index+", Size: "+size);

}

不用解释。

private synchronized void writeObject(java.io.ObjectOutputStream s)throws java.io.IOException{

// Write out element count, and any hidden stuff

s.defaultWriteObject();

// Write out array length

s.writeInt(elementData.length);

// Write out all elements in the proper order.

for (int i=0; i

s.writeObject(elementData[i]);

}

private synchronized void readObject(java.io.ObjectInputStream s)throws java.io.IOException,ClassNotFoundException {

// Read in size, and any hidden stuff

s.defaultReadObject();

// Read in array length and allocate array

int arrayLength = s.readInt();

elementData = new Object[arrayLength];

// Read in all elements in the proper order.

for (int i=0; i

elementData[i] = s.readObject();

}

这两个方法是为了实现串行化而写的,这里要注意几点:

1.这两个方法并不是java.io.Serializable中定义的方法,Serializable只是标记接口。

2.串行化对象的时候会先检查该对象是否实现了这两个方法,如果有实现就调用他们,如果没有的化就调用默认方法。

至于为什么可以调用该对象的private方法我也不清楚,这个问题确实比较奇怪,如果可以访问对象的private方法,那

也太不安全了。

3.因为elementData声明为transient,所以必须手动串行化化。

总结:

1.ArrayList的方法都没有同步,所以在多线程中是不安全的,必须自己同步,而且ArrayList很多方法声明对于多线程操作

都没有规定,就是说结果不可预料。

2.toArray()方法返回的是和原列表相同的对象,也就是说:

arrayList.toArray()[0]==arrayList.get(0)返回的是true(假定arrayList不空)。

3.clone()方法是一个浅拷贝。

4.可以通过自己调用ensureCapacity()提高效率。

 
 
 
免责声明:本文为网络用户发布,其观点仅代表作者个人观点,与本站无关,本站仅提供信息存储服务。文中陈述内容未经本站证实,其真实性、完整性、及时性本站不作任何保证或承诺,请读者仅作参考,并请自行核实相关内容。
2023年上半年GDP全球前十五强
 百态   2023-10-24
美众议院议长启动对拜登的弹劾调查
 百态   2023-09-13
上海、济南、武汉等多地出现不明坠落物
 探索   2023-09-06
印度或要将国名改为“巴拉特”
 百态   2023-09-06
男子为女友送行,买票不登机被捕
 百态   2023-08-20
手机地震预警功能怎么开?
 干货   2023-08-06
女子4年卖2套房花700多万做美容:不但没变美脸,面部还出现变形
 百态   2023-08-04
住户一楼被水淹 还冲来8头猪
 百态   2023-07-31
女子体内爬出大量瓜子状活虫
 百态   2023-07-25
地球连续35年收到神秘规律性信号,网友:不要回答!
 探索   2023-07-21
全球镓价格本周大涨27%
 探索   2023-07-09
钱都流向了那些不缺钱的人,苦都留给了能吃苦的人
 探索   2023-07-02
倩女手游刀客魅者强控制(强混乱强眩晕强睡眠)和对应控制抗性的关系
 百态   2020-08-20
美国5月9日最新疫情:美国确诊人数突破131万
 百态   2020-05-09
荷兰政府宣布将集体辞职
 干货   2020-04-30
倩女幽魂手游师徒任务情义春秋猜成语答案逍遥观:鹏程万里
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案神机营:射石饮羽
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案昆仑山:拔刀相助
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案天工阁:鬼斧神工
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案丝路古道:单枪匹马
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案镇郊荒野:与虎谋皮
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案镇郊荒野:李代桃僵
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案镇郊荒野:指鹿为马
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案金陵:小鸟依人
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案金陵:千金买邻
 干货   2019-11-12
 
推荐阅读
 
 
 
>>返回首頁<<
 
靜靜地坐在廢墟上,四周的荒凉一望無際,忽然覺得,淒涼也很美
© 2005- 王朝網路 版權所有