ArrayList原理

一、定义

ArrayList是android开发应用最多的一种集合,在处理比较复杂的业务时候,需要遍历来查找数据和操作数据的时候都会考虑用ArrayList去实现。

二、介绍

1,ArrayList的原理跟javaScript类似都是基于数组实现的,这样可以节约空间,众所周知数组是有容量限制,因此它是一个动态数组,超出限制时会自动增加原来的1/2(50%)容量,所以其容量能自动增长,即自动扩容机制,如果可预知数据量的多少,可在构造ArrayList时指定其容量。

2,实现所有可选的列表操作,并允许所有元素,包括null。

3,提供一些方法来操作内部用来存储列表的数组的大小(这类大致相当于载体,但它是不同步的)

三、缺点

非线程安全,只能用在单线程环境下,不能适应数据动态地增减的情况(当数据增加时,可能超出原先定义的元素个数;当数据减少时,造成内存浪费)。

四、实现方式

1,构造函数

//可以构造一个默认初始容量为10的空列表

public ArrayList() {

this(10);

}

//构造一个指定初始容量的空列表

public ArrayList(int initialCapacity) {

super();

if(initialCapacity <0)

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

this.elementData =new Object [initialCapacity];

}

//构造一个包含指定collection的元素的列表,这些元素按照该collection的迭代器返回它们的顺序排列的。

public ArrayList(Collection c) {

elementData = c.toArray();

size = elementData.length;

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

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

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

}

2,储存方式

// 用指定的元素替代此列表中指定位置上的元素,并返回以前位于该位置上的元素。

public E set(int index, E element) {

RangeCheck(index);

EoldValue = (E) elementData[index];

elementData[index] = element;

return oldValue;

}

// 将指定的元素添加到此列表的尾部。

public boolean add(E e) {

ensureCapacity(size + 1);

elementData[size++] = e;

return true;

}

// 将指定的元素插入此列表中的指定位置。如果当前位置有元素,则向右移动当前位于该位置的元素以及所有后续元素(将其索引加1)。

public void add(int index, E element) {

if (index > size || index < 0)

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

// 如果数组长度不足,将进行扩容。

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

// 将 elementData中从Index位置开始、长度为size-index的元素,

// 拷贝到从下标为index+1位置开始的新的elementData数组中。

// 即将当前位于该位置的元素以及所有后续元素右移一个位置。

System.arraycopy(elementData, index, elementData, index + 1, size - index);

elementData[index] = element;

size++;

}

// 按照指定collection的迭代器所返回的元素顺序,将该collection中的所有元素添加到此列表的尾部。

public boolean addAll(Collection c) {

Object[] a = c.toArray();

int numNew = a.length;

ensureCapacity(size + numNew);  // Increments modCount

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

size += numNew;

return numNew != 0;

}

// 从指定的位置开始,将指定collection中的所有元素插入到此列表中。

public boolean addAll(int index, Collection c) {

if (index > size || index < 0)

throw new IndexOutOfBoundsException(

"Index: " + index + ", Size: " + size);

Object[] a = c.toArray();

int numNew = a.length;

ensureCapacity(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;

}

3, 元素删除:

// 移除此列表中指定索引上的元素。

public E remove(int index) {

RangeCheck(index);

modCount++;

E oldValue = (E) 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;

}

4,元素更改(这一栏本身是不需要的,只不过对数据操作离不开增删改查,所以单独的在列出来)

// 用指定的元素替代此列表中指定位置上的元素,并返回以前位于该位置上的元素。

public E set(intindex, E element) {

RangeCheck(index);

E oldValue = (E) elementData[index];

elementData[index] = element;returnoldValue;

}

5,元素读取

// 返回此列表中指定索引上的元素。

public E get(int index) {

RangeCheck(index);

return (E) elementData[index];

}

6,调整数据容器

//在实际添加大量元素前,咱们也可以使用ensureCapacity来手动增加ArrayList实例的容量,以减少递增式再分配的数量。

public void ensureCapacity(int minCapacity) {

if(minCapacity > oldCapacity) {

Object oldData[] = elementData;

intnewCapacity = (oldCapacity *3) /2+1;

if(newCapacity < minCapacity)

newCapacity = minCapacity;

elementData = Arrays.copyOf(elementData, newCapacity);

}

}

//将底层数组的容量调整为当前列表保存的实际元素的大小的功能。

public void trimToSize() {

modCount++;

int oldCapacity = elementData.length;

if (size < oldCapacity) {

elementData = Arrays.copyOf(elementData, size);

}

}

综上两种方法

ensureCapacity方法中elementData的长度会被拓展,size标记的是其中包含的元素的个数。所以会出现size很小但elementData.length很大的情况,将出现空间的浪费。

trimToSize将返回一个新的数组elementData,元素内容保持不变,length和size相同,节省空间。

总结:当然 ArrayList  里面的方法肯定不止这些,咱们只是把常用的方法给展示出来了。 在实际使用过程中。对ArrayList使用要注意以下几点:

1, 初始化ArrayList的时候,尽量指定其容量大小 。

2, 当操作完一个列表后,使用 trimToSize 方法去确认长度,不然可能会有容量浪费。

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

推荐阅读更多精彩内容