ArrayList
各种原因,前两年做C语言去了,现在重新做JAVA, 感觉自己基础很不扎实,要好好学习啦, 先从简单的开始~
以下内容基于jdk1.7.0_79源码;
什么是ArrayList
可以简单的认为是一个动态数组;实际上ArrayList就是用数组实现的,长度不够时,调用Arrays.copyOf方法,拷贝当前数组到一个新的长度更大的数组;
ArrayList特点
随机访问速度快,插入和移除性能较差(数组的特点);
支持null元素;
有顺序;
元素可以重复;
线程不安全;
ArrayList继承的类和实现的接口
如下图,是与ArrayList相关的接口和类,下面将一一介绍各个接口和类中的方法;
PS:ArrayList中的方法主要是由Collection接口和List接口定义的;
Iterable接口
实现此接口以便支持foreach语法,如下代码,ArrayList可以直接使用foreach语句遍历元素:
package com.pichen.basis.col; import java.util.ArrayList;
import java.util.List; public class Main { public static void main(String[] args) {
List<Integer> list = new ArrayList<Integer>();
for(int i = 0; i < 10; i++){
list.add(i);
} //foreach语法
for(Integer i : list){
System.out.print(i.toString() + " ");
} }
}
Collection接口
int size()方法:
返回集合的大小,在ArrayList类中有一个int类型的size私有属性,当调用size方法时,直接返回该属性;
boolean isEmpty()方法:
判断集合是否为空,在ArrayList中,通过判断size == 0来判断集合是否为空;
boolean contains(Object o)方法:
判断集合是否含有对象o,在ArrayList中,通过判断indexOf(o) >= 0来判断是否含有o对象;
查看indexOf(o)方法,代码如下,主要功能是返回元素第一次出现时的下标索引,所以当下标索引大于等于0时,表示集合中存在该元素:
public int indexOf(Object o) {
if (o == null) {
for (int i = 0; i < size; i++)
if (elementData[i]==null)
return i;
} else {
for (int i = 0; i < size; i++)
if (o.equals(elementData[i]))
return i;
}
return -1;
}
注意这里的相等判断,调用的是o对象的equals方法,所以在调用contains方法时,要特别关注集合中对象的equals方法,是否有被重写过,如Integer、String的equals方法是被重写过的,一般我们自己定义的对象,如果没重写equals的话,默认调用的是Object的equals方法,举个例子,看一下就明白了:
package com.pichen.basis.col; import java.util.ArrayList;
import java.util.List; class Dog{
}
public class ContainsTest { public static void main(String[] args) {
List<Dog> dogList = new ArrayList<Dog>();
Dog dog1 = new Dog();
Dog dog2 = new Dog();
dogList.add(dog1);
System.out.println(dogList.contains(dog2));//false List<String> strList = new ArrayList<String>();
strList.add("teststr"); String str = new String("teststr");
System.out.println(strList.contains(str));//true } }
Iterator<E> iterator()方法:
返回一个迭代器对象,用于遍历集合,事实上,ArrayList类里有两个内部类ArrayList.Itr和ArrayList.ListItr,分别对应Iterator迭代器和ListIterator迭代器,后者比前者功能更加强大;从ArrayList.ListItr继承自ArrayList.Itr就可以看出来,ListIterator迭代器支持更多的操作,如判断前面还有没有元素,即hasPrevious()方法,等;
Object[] toArray()方法:
将集合ArrayList转换成Object数组,有时候需要用到数组的一些api时,可以使用该方法,注意返回的结果是Object类型的数组,如果想返回指定类型的数组,可以使用以下方法,<T> T[] toArray(T[] a);
<T> T[] toArray(T[] a)方法:
集合转数组,返回指定类型的数组,注意入参T[] a需要指定数组存储的空间,返回值为指定类型的数组;举个例子,假如有一个Integer类型的集合,如果想把它转换成Integer类型的数组,可以这样写:Integer[] arr = list.toArray(new Integer[list.size()]);
boolean add(E e)方法:
在集合最后面增加一个元素,在ArrayList中,其实现就是在其内部数组后面增加一个元素,不过要先保证内部数组长度足够,如下代码:
public boolean add(E e) {
ensureCapacityInternal(size + 1); // Increments modCount!!
elementData[size++] = e;
return true;
}
boolean remove(Object o)方法:
在集合中移除对象o,在ArrayList中,其实现较add方法复杂,涉及空对象判断,equals比较,数组移动等,性能相对较差;
boolean containsAll(Collection<?> c)方法:
判断是否包含集合c中的所有元素,在ArrayList中,其实现方法是遍历集合c中的每一个元素,调用contains方法,判断集合是否包含该元素,只要有一个不包含就返回false,如下代码:
public boolean containsAll(Collection<?> c) {
for (Object e : c)
if (!contains(e))
return false;
return true;
}
boolean addAll(Collection<? extends E> c)方法:
将集合c中的所有元素加到目标集合中去,在ArrayList中,其实现是先将集合c转换成数组,然后通过数组拷贝实现;
boolean removeAll(Collection<?> c)方法:
移除目标集合中含有‘集合c中元素’的所有元素,在ArrayList中,最终还是操作数组,性能相对较差;
boolean retainAll(Collection<?> c)方法:
移除目标集合中‘不包含集合c中元素’的所有元素,在ArrayList中removeAll方法和retainAll方法都是通过调用ArrayList的batchRemove方法来实现的,后续详细了解该方法的实现;
void clear()方法:
移除目标集合中的所有元素,在ArrayList中,就是将其内部数组所有元素赋null;
boolean equals(Object o)和int hashCode()方法
在ArrayLisy中,上面两个方法都被重写,equals方法依次取出集合中的所有元素进行比较,通过元素的equals方法,判断是否相等,全部相等返回true;
hashCode方法的计算是通过所有元素的hashCode计算得到;顺便说下hashcode,在java中随处可见,一般用在HashMap, Hashtable, HashSet等等中,可用于减少equals方法的调用,快速访问元素等,其实就是散列表的概念,如比较元素先比较其hashcode,如果hashcode不相等,那么这两个元素肯定不相等,也就不用调用其equals方法了;
demo代码:以上方法的简单使用
package com.pichen.basis.col; import java.util.ArrayList;
import java.util.Arrays;
import java.util.Iterator;
import java.util.List; public class Test { public static void main(String[] args) {
List<Integer> list = new ArrayList<Integer>();
for(int i = 0; i < 10; i++){
list.add(i);
} //直接打印
System.out.println(list.toString()); //size()
System.out.println(list.size()); //contains
System.out.println(list.contains(2)); //iterator
Iterator<Integer> iterator = list.iterator();
while(iterator.hasNext()){
System.out.print(iterator.next() + " ");
} //toArray
Object[] objArr = list.toArray();
Integer[] intArr = list.toArray(new Integer[list.size()]);
System.out.println("\n" + list.toArray()); //add
list.add(5); //remove
list.remove(5); System.out.println(list); //containsAll
System.out.println(list.containsAll(Arrays.asList(5,6))); //addAll
list.addAll(Arrays.asList(555,666));
System.out.println(list); //removeAll
list.removeAll(Arrays.asList(555,666));
System.out.println(list); //clear
list.clear();
System.out.println(list); }
}
List接口
除了Collection中定义的方法为,该接口增加了以下方法
boolean addAll(int index, Collection<? extends E> c);
在ArrayList中,该方法是在指定位置处增加一个集合中的所有元素,该操作涉及数组移动;
E get(int index);
返回下标为index的元素;
E set(int index, E element);
改变下标为index的元素的值
void add(int index, E element);
在下标为index的地方插入元素element,该操作涉及数组移动;
E remove(int index);
移除下标为index的元素,该操作涉及数组移动;
int indexOf(Object o);
返回元素o的最小下标,通过调用o的equals方法与集合中的元素进行比较;
int lastIndexOf(Object o);
返回元素o的最大下标,通过调用o的equals方法与集合中的元素进行比较;
ListIterator<E> listIterator();
返回listIterator迭代器,该迭代器支持向前操作;
ListIterator<E> listIterator(int index);
返回listIterator迭代器,从特定的位置开始,该迭代器支持向前操作;
List<E> subList(int fromIndex, int toIndex);
返回下标在fromIndex和toIndex之间的元素集合;
demo代码:以上方法的简单使用:
package com.pichen.basis.col; import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;
import java.util.ListIterator; public class Main { public static void main(String[] args) {
List<Integer> list = new ArrayList<Integer>();
for(int i = 0; i < 10; i++){
list.add(i);
} System.out.println(list); //addAll方法
list.addAll(5, Arrays.asList(666,666, 6));
System.out.println(list); //get方法
System.out.println(list.get(5)); //set方法
list.set(5, 55);
System.out.println(list.get(5)); //add方法
list.add(0, 555);
System.out.println(list); //remove方法
list.remove(0);
System.out.println(list); //indexof方法
System.out.println(list.indexOf(6)); //lastIndexOf方法
System.out.println(list.lastIndexOf(6)); //listIterator方法
ListIterator<Integer> listIterator = list.listIterator();
System.out.println(listIterator.hasPrevious()); //listIterator(index)方法
ListIterator<Integer> iListIterator = list.listIterator(5);
System.out.println(iListIterator.previous()); //subList方法
System.out.println(list.subList(5, 7));
}
}
RandomAccess, Cloneable, java.io.Serializable接口
这三个接口是标识接口,里面都是空的;
RandomAccess标识其支持快速随机访问;
Cloneable标识其支持对象复制;
Serializable标识其可序列化;
AbstractCollection类
大部分方法前面已经说明过了,不过该类下的contains方法、toArray方法等,遍历的时候都是使用更加通用的迭代器方式进行遍历;
AbstractList类
大部分方法前面已经说明过了,不过该类中有两个私有内部类Itr和ListItr,对应的分别是两个迭代器;
ArrayList类
ArrayList的具体实现
成员属性:
private static final int DEFAULT_CAPACITY = 10;//初始容量
private static final Object[] EMPTY_ELEMENTDATA = {};//空ArrayList实例共享的一个空数组
private transient Object[] elementData; //真正存储ArrayList中的元素的数组;
private int size;//存储ArrayList的大小,注意不是elementData的长度;
除了其父接口定义的方法外,该类增加了以下方法
public ArrayList(int initialCapacity):
构造函数,指定初始大小
public ArrayList()
构造函数,使用共享的EMPTY_ELEMENTDATA空数组
public ArrayList(Collection<? extends E> c)
构造函数,通过集合初始化ArrayList
public void trimToSize()
节省空间用的,ArrayList是通过数组实现的,大小不够时,增加数组长度,有可能出现数组长度大于ArrayList的size情况;
public void ensureCapacity(int minCapacity)
保证ArrayList能容纳minCapacity个元素;
私有方法
略
ArrayList内部数组扩容
查看ArrayList的add源码方法,如下:
public boolean add(E e) {
ensureCapacityInternal(size + 1); // Increments modCount!!
elementData[size++] = e;
return true;
}
在往ArrayList增加e对象前,要先保证内部数组空间足够,即调用ensureCapacityInternal判断,查看其代码,如下:
private void ensureCapacityInternal(int minCapacity) {
if (elementData == EMPTY_ELEMENTDATA) {
minCapacity = Math.max(DEFAULT_CAPACITY, minCapacity);
} ensureExplicitCapacity(minCapacity);
}
入参minCapacity为ArrayList要保证的最小空间大小,首先判断内部数组elementData是否是初始的空数组(当ArrayList是空实例的情况),如果是的话,且minCapacity比DEFAULT_CAPACITY小,则设minCapacity为默认的初始容量大小DEFAULT_CAPACITY;接着调用ensureExplicitCapacity方法,查看其代码,如下:
private void ensureExplicitCapacity(int minCapacity) {
modCount++; // overflow-conscious code
if (minCapacity - elementData.length > 0)
grow(minCapacity);
}
首先,是对修改次数modCount加一,这里的modCount给ArrayList的迭代器使用的,在并发操作被修改时,提供快速失败行为(保证modCount在迭代期间不变,否则抛出ConcurrentModificationException异常,可以查看源码859行),接着判断minCapacity是否大于当前ArrayList内部数组长度,大于的话调用grow方法对内部数组elementData扩容,grow方法代码如下:
private void grow(int minCapacity) {
// overflow-conscious code
int oldCapacity = elementData.length;
int newCapacity = oldCapacity + (oldCapacity >> 1);
if (newCapacity - minCapacity < 0)
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);
}
首先,获取内部数组elementData长度,并对它的大小增加1.5倍(oldCapacity + (oldCapacity >> 1)),赋给newCapacity,当newCapacity仍然比minCapacity(要保证的最小空间大小)小的时候,直接让newCapacity 等于minCapacity;然后再判断newCapacity 是不是大于MAX_ARRAY_SIZE,如果大于的话,调用hugeCapacity方法,处理大容量空间的情况,该方法判断是否溢出(抛出OutOfMemoryError),以及当minCapacity大于MAX_ARRAY_SIZE时,直接返回Integer.MAX_VALUE;最后就是真正地数组扩容了,调用Arrays.copyOf方法,拷贝旧的内部数组内容到一个新的(长度为newCapacity)的数组,并更改内部数组elementData的引用为新数组;
ArrayList遍历
一般工作中用的比较多的是遍历操作,ArrayList支持三种方式
- for循环下标遍历;
- 迭代器(Iterator和ListIterator);
- foreach语句。
package com.pichen.basis.col; import java.util.ArrayList;
import java.util.Iterator;
import java.util.List;
import java.util.ListIterator; public class Test { public static void main(String[] args) {
List<Integer> list = new ArrayList<Integer>();
for(int i = 0; i < 10; i++){
list.add(i);
} //直接打印
System.out.println(list.toString()); //for循环
System.out.println("for循环:");
for(int i = 0; i < list.size(); i++){
System.out.print(list.get(i) + " ");
} //foreach
System.out.println("\nforeach:");
for(Integer i : list){
System.out.print(i + " ");
} //iterator
System.out.println("\niterator:");
Iterator<Integer> iterator = list.iterator();
while(iterator.hasNext()){
System.out.print(iterator.next() + " ");
} //listIterator
System.out.println("\nlistIterator:");
ListIterator<Integer> listIterator = list.listIterator();
while(listIterator.hasNext()){
System.out.print(listIterator.next() + " ");
}
System.out.println();
while(listIterator.hasPrevious()){
System.out.print(listIterator.previous() + " ");
}
}
}
ArrayList的更多相关文章
- 计算机程序的思维逻辑 (38) - 剖析ArrayList
从本节开始,我们探讨Java中的容器类,所谓容器,顾名思义就是容纳其他数据的,计算机课程中有一门课叫数据结构,可以粗略对应于Java中的容器类,我们不会介绍所有数据结构的内容,但会介绍Java中的主要 ...
- 分享个 之前写好的 android 文件流缓存类,专门处理 ArrayList、bean。
转载麻烦声明出处:http://www.cnblogs.com/linguanh/ 目录: 1,前序 2,作用 3,特点 4,代码 1,前序 在开发过程中,client 和 server 数据交流一 ...
- 深入理解java中的ArrayList和LinkedList
杂谈最基本数据结构--"线性表": 表结构是一种最基本的数据结构,最常见的实现是数组,几乎在每个程序每一种开发语言中都提供了数组这个顺序存储的线性表结构实现. 什么是线性表? 由0 ...
- 【干货】用大白话聊聊JavaSE — ArrayList 深入剖析和Java基础知识详解(二)
在上一节中,我们简单阐述了Java的一些基础知识,比如多态,接口的实现等. 然后,演示了ArrayList的几个基本方法. ArrayList是一个集合框架,它的底层其实就是一个数组,这一点,官方文档 ...
- WebAPI接口返回ArrayList包含Dictionary对象正确解析
一.问题提出 为了减少流量,将key-value(键值对)直接输出到Dictionary<string, string>,接口返回结果如下: 其中{}里面内容如下: 上图显示600是键,4 ...
- ArrayList LinkedList源码解析
在java中,集合这一数据结构应用广泛,应用最多的莫过于List接口下面的ArrayList和LinkedList; 我们先说List, public interface List<E> ...
- ArrayList、Vector、LinkedList的区别联系?
1.ArrayList.Vector.LinkedList类都是java.util包中,均为可伸缩数组. 2.ArrayList和Vector底层都是数组实现的,所以,索引数据快,删除.插入数据慢. ...
- ArrayList、Vector、HashMap、HashSet的默认初始容量、加载因子、扩容增量
当底层实现涉及到扩容时,容器或重新分配一段更大的连续内存(如果是离散分配则不需要重新分配,离散分配都是插入新元素时动态分配内存),要将容器原来的数据全部复制到新的内存上,这无疑使效率大大降低. 加载因 ...
- Java中Vector和ArrayList的区别
首先看这两类都实现List接口,而List接口一共有三个实现类,分别是ArrayList.Vector和LinkedList.List用于存放多个元素,能够维护元素的次序,并且允许元素的重复.3个具体 ...
- C#语言基础——集合(ArrayList集合)
集合及特殊集合 集合的基本信息: System.Collections 命名空间包含接口和类,这些接口和类定义各种对象(如列表.队列.位数组.哈希表和字典)的集合.System.Collections ...
随机推荐
- 一些sql二
1.说明:复制表(只复制结构,源表名:a 新表名:b) (Access可用)法一:select * into b from a where 1<>1(仅用于SQlServer)法二:sel ...
- mvc 修改 删除 linq
Models文件夹里面可以Linq,Entity两种映射框架,也允许有ADO的操作,甚至可以ADO代码和映射代码一起操作 控制器当中允许有相同的方法名,类似在做修改时有两个Modify方法,但是MVC ...
- SSH框架执行自己定义的SQL语句
直接上代码 String hsql = "delete XTable x where x.Userid= ?"; Query query = this.getSession().c ...
- 使用Eclipse为Android定义style
1.首先,在values目录下,新建一个styles.xml文件: 2.进入styles.xml文件,点击Resources: 3.点击Add按钮,在弹出的对话框中选择在顶层创建新元素,在选择Styl ...
- PLSQL怎样导出oracle表结构和数据
1.导出表结构和数据方式1.tools->export user objects是导出表结构 tools ->export user object 选择选项,导出.sql文件 说明:导出的 ...
- HTML 限制文本框只能输入特定字符(比如数字 onkeyup+onafterpaste)
正则表达式1 <td><asp:TextBox ID="TextBox_username" Width="250" runat="s ...
- activiti工作流的web流程设计器整合视频教程
本视频为activiti工作流的web流程设计器整合视频教程 整合Acitiviti在线流程设计器(Activiti-Modeler 5.21.0 官方流程设计器) 本视频共讲了两种整合方式 1. 流 ...
- 粒子动画Particleground.js
<!DOCTYPE html> <html> <head> <meta charset="UTF-8"> <title> ...
- jquery.datepair日期时分秒选择器
jquery.datepair是一个轻量级的jQuery插件,智能选择日期和时间范围,灵感来自于谷歌日历.Datepair将保持开始和结束日期/时间同步,并可以根据用户的操作设置默认值.该插件不提供任 ...
- 太可爱了!CSS3 & SVG 制作的米老鼠钟表
米老鼠是大家非常熟悉的迪斯尼动画形象.这是一个可爱的效果,结合 CSS & SVG 图形实现的米老鼠钟表效果.Web 技术让很多生活中的事物都能搬到网上去,后面的推荐阅读也有很多的效果,感兴趣 ...