1. ArrayList
A) 底层数据结构
·本质是一个Object数组,存放的是对象引用序列。size代表元素个数。
·采用数组并通过算法保证了集合元素有序,允许重复的特性。
B) 构造方法
·创建一个ArrayList,默认大小为10。
C) 插入对象
·插入元素面临的一个重要的问题就是空间不够(这也是数组的最大弊端),如何扩容?ArrayList是通过一个公开的方法ensureCapacity(int minCapacity)来实现。
·先计算新的数组大小
·原数组的容量*1.5+1
·扩容后会检查是否能满足需要,若数组大小还不够,会直接扩容到需要的大小。这种情况主要发生在批量增加集合中的元素。
·再扩容(按照新的大小生成数组),同时拷贝原ArrayList中的元素。
·底层调用的是native方法:System.arraycopy(original, 0, copy, 0, Math.min(original.length, newLength));
·频繁的扩容(移动元素)是耗性能的。若能明确List大小,给ArrayList设置合理的初始值是比较理想的。
·然后将新增元素插入到指定位置。
·批量增加集合中的元素会涉及两次拷贝。扩容会拷贝原数组元素,还会拷贝需要增加的集合中的元素。
D)删除对象
·先通过for循环遍历(三种遍历方式:for循环,foreach,迭代iterator)找到删除元素。
·然后移动元素(被删除元素后的所有元素),再将最后一个元素设置为null,交给GC完成对象的回收。
·对于负数是不检查的,而是直接访问,直到抛出ArrayIndexOutOfBoundsException,参考 RangeCheck(int index)。
·删除元素,但是数组空间不会释放,可以调用trimToSize()缩小数组容量(ArrayList不会自动调用)
E)查找对象
·获取对象的位置:indexOf(Object o),从第一个元素往后查找;lastIndexOf(Object o),从最后一个元素往前查找。
·判断对象是否存在:contains(Object o),本质调用的是indexOf(Object o)。
·采用for循环遍历查找的方式。
2.LinkedList
A)底层数据结构
·底层采用双向循环链表结构实现。
·header:双向链表的头;Entry:包含三部分--对象数据,指向后一个Entry的引用,指向前一个Entry的引用。
B)构造方法
·构造Entry(null,null,null):属性值全为null,同时赋值给header。
·构造方法仅仅生成双向链表结构:header.next = header.previous = header。
C)插入对象
·不用考虑扩容和数据复制(移动)。
·在header前插入,无须遍历。
·创建一个新对象,修改相邻元素属性。
D)删除对象
·遍历与匹配和ArrayList相似。
·删除元素:修改相邻元素属性,被删除元素属性置为null。无需复制元素。
E)查找对象
·与ArrayList相似,不同的是LinkedList不支持下标index,在遍历的时候,临时记录一个index。
·获取对象的位置:indexOf(Object o),从第一个元素往后查找;lastIndexOf(Object o),从最后一个元素往前查找。
·判断对象是否存在:contains(Object o),本质调用的是indexOf(Object o)。
·采用for循环遍历查找的方式。
3.Vector
·继承AbstractList,实现List接口。与ArrayList相似。
·线程安全。
·初始大小为10。
·扩容策略与ArrayList不一样:
·通过capacityIncrement控制:如果capacityIncrement>0,每次增加capacityIncrement。否则扩大为原来的两倍。
4.Stack
·后进先出:LIFO;入栈:push(E);出栈:pop(),返回最后一个元素并删除元素;peek(),返回最后一个元素但不删除。
·继承Vector,线程安全的,这使得stack也变得重量级。
·stack的父类不应该为Vector的,因为Vector的底层是数组(增删效率比较低),且Vector有get方法(意味着它可能访问到并不属于最后一个位置元素的其他元素,很不安全)。
·不考虑并发,怎么实现Stack的功能?封装LinkedList也许是不错的选择。
A) 底层数据结构
·本质是一个Object数组,存放的是对象引用序列。size代表元素个数。
·采用数组并通过算法保证了集合元素有序,允许重复的特性。
B) 构造方法
public ArrayList() { this(10); }
·创建一个ArrayList,默认大小为10。
C) 插入对象
·插入元素面临的一个重要的问题就是空间不够(这也是数组的最大弊端),如何扩容?ArrayList是通过一个公开的方法ensureCapacity(int 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.5+1
·扩容后会检查是否能满足需要,若数组大小还不够,会直接扩容到需要的大小。这种情况主要发生在批量增加集合中的元素。
·再扩容(按照新的大小生成数组),同时拷贝原ArrayList中的元素。
·底层调用的是native方法:System.arraycopy(original, 0, copy, 0, Math.min(original.length, newLength));
·频繁的扩容(移动元素)是耗性能的。若能明确List大小,给ArrayList设置合理的初始值是比较理想的。
·然后将新增元素插入到指定位置。
·批量增加集合中的元素会涉及两次拷贝。扩容会拷贝原数组元素,还会拷贝需要增加的集合中的元素。
public boolean addAll(Collection<? extends E> 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; }
D)删除对象
private void fastRemove(int index) { modCount++; int numMoved = size - index - 1; if (numMoved > 0) System.arraycopy(elementData, index+1, elementData, index, numMoved); elementData[--size] = null; // Let gc do its work }
·先通过for循环遍历(三种遍历方式:for循环,foreach,迭代iterator)找到删除元素。
·然后移动元素(被删除元素后的所有元素),再将最后一个元素设置为null,交给GC完成对象的回收。
·对于负数是不检查的,而是直接访问,直到抛出ArrayIndexOutOfBoundsException,参考 RangeCheck(int index)。
·删除元素,但是数组空间不会释放,可以调用trimToSize()缩小数组容量(ArrayList不会自动调用)
public void trimToSize() { modCount++; int oldCapacity = elementData.length; if (size < oldCapacity) { elementData = Arrays.copyOf(elementData, size); } }
E)查找对象
·获取对象的位置:indexOf(Object o),从第一个元素往后查找;lastIndexOf(Object o),从最后一个元素往前查找。
·判断对象是否存在:contains(Object o),本质调用的是indexOf(Object o)。
·采用for循环遍历查找的方式。
2.LinkedList
A)底层数据结构
·底层采用双向循环链表结构实现。
·header:双向链表的头;Entry:包含三部分--对象数据,指向后一个Entry的引用,指向前一个Entry的引用。
B)构造方法
public LinkedList() { header.next = header.previous = header; } private transient Entry<E> header = new Entry<E>(null, null, null); Entry(E element, Entry<E> next, Entry<E> previous) { this.element = element; this.next = next; this.previous = previous; }
·构造Entry(null,null,null):属性值全为null,同时赋值给header。
·构造方法仅仅生成双向链表结构:header.next = header.previous = header。
C)插入对象
private Entry<E> addBefore(E e, Entry<E> entry) { Entry<E> newEntry = new Entry<E>(e, entry, entry.previous); newEntry.previous.next = newEntry; newEntry.next.previous = newEntry; size++; modCount++; return newEntry; }
·不用考虑扩容和数据复制(移动)。
·在header前插入,无须遍历。
·创建一个新对象,修改相邻元素属性。
D)删除对象
·遍历与匹配和ArrayList相似。
·删除元素:修改相邻元素属性,被删除元素属性置为null。无需复制元素。
E)查找对象
·与ArrayList相似,不同的是LinkedList不支持下标index,在遍历的时候,临时记录一个index。
·获取对象的位置:indexOf(Object o),从第一个元素往后查找;lastIndexOf(Object o),从最后一个元素往前查找。
·判断对象是否存在:contains(Object o),本质调用的是indexOf(Object o)。
·采用for循环遍历查找的方式。
3.Vector
·继承AbstractList,实现List接口。与ArrayList相似。
·线程安全。
·初始大小为10。
·扩容策略与ArrayList不一样:
private void ensureCapacityHelper(int minCapacity) { int oldCapacity = elementData.length; if (minCapacity > oldCapacity) { Object[] oldData = elementData; int newCapacity = (capacityIncrement > 0) ? (oldCapacity + capacityIncrement) : (oldCapacity * 2); if (newCapacity < minCapacity) { newCapacity = minCapacity; } elementData = Arrays.copyOf(elementData, newCapacity); } }
·通过capacityIncrement控制:如果capacityIncrement>0,每次增加capacityIncrement。否则扩大为原来的两倍。
4.Stack
·后进先出:LIFO;入栈:push(E);出栈:pop(),返回最后一个元素并删除元素;peek(),返回最后一个元素但不删除。
·继承Vector,线程安全的,这使得stack也变得重量级。
·stack的父类不应该为Vector的,因为Vector的底层是数组(增删效率比较低),且Vector有get方法(意味着它可能访问到并不属于最后一个位置元素的其他元素,很不安全)。
·不考虑并发,怎么实现Stack的功能?封装LinkedList也许是不错的选择。
发表评论
-
[转载]Java注解--源码解析
2012-04-24 18:59 2455注解提供了一种结构化的,并且具有类型检查能力的新途径,从而使程 ... -
J2EE、J2SE、J2ME区别
2012-04-21 18:07 1366JAVA2平台是提供JAVA程序开发、运行环境的平台,JAVA ... -
[转载]JDK和JRE目录的文件结构
2012-04-21 17:12 1885[转载 ] 我们下 ... -
[转载]SDK、JDK、JRE和JVM的关系总结
2012-04-12 22:16 2064一、SDK、JDK、JRE和JVM的 ... -
Java注解
2012-04-11 02:02 1851可以先看看转载的三篇博客: Java注解--基础知识 ... -
[转载]Java注解--基础知识
2012-04-10 23:53 1528[转载 ] 一、什么是java 注 ... -
[转载]Java注解--原理
2012-04-10 23:34 1271[转载 ] 在开发Java ... -
集合初探--集合中的其它设计模式
2011-03-27 21:35 12571.集合中的工厂方法模式 ·工厂方法(FactoryMet ... -
集合初探--集合中的设计模式之Iterator模式
2011-03-27 21:35 12951. Iterator模式 ·标准定义:提供一种统一的方法顺 ... -
集合初探--Fail-Fast机制
2011-03-27 21:35 1211Fail-Fast机制 ·在系统发生错误后,立即作出响应,阻 ... -
集合初探--认识Set
2011-03-27 21:34 10221. HashSet ·基于HashMap实现的,Hash ... -
集合初探--认识Map
2011-03-27 21:34 11211. HashMap A)底层数据结构 ·HashMap ... -
集合初探--集合框架
2011-03-24 09:44 1132最近学习了java集合,将自己学习的笔记整理后发布到博客,本系 ...
相关推荐
Android中滑屏初探 ---- scrollTo 以及 scrollBy方法使用说明 Android中滑屏初探 ---- scrollTo 以及 scrollBy方法使用说明 Android中滑屏初探 ---- scrollTo 以及 scrollBy方法使用说明
《ODI_11G初探-简单数据传输》这一文档深入探讨了Oracle Data Integrator (ODI) 11G版本在数据传输中的应用与配置过程,为初学者提供了全面而详细的指导。ODI是Oracle公司推出的一款企业级数据集成工具,用于实现...
Springboot初探---FreeMarker 之 HelloWorld,很好的资源
广播文化类线性节目”本土化”初探------以陕西交通广播”长安处处有故事”为例.zip
广播文化类线性节目”本土化”初探------以陕西交通广播”长安处处有故事”为例.doc
人事档案社会化管理初探--也谈人档分离-论文.zip
c语言程序设计教材建设初探-程序设计-设计.pdf
小程序在博物馆展览中的潜力初探--以故宫博物院端门数字馆导览小程序为例
医院消防安全管理初探-安全管理-行业安全-消防安全.docx
90年代至新世纪我国谈话类节目 主广播文化类线性节目”本土化”初探------以陕西交通广播”长安处处有故事”为例持风格的演进.zip
国内互联网保险初探-论文.zip
《基于GDI+》2D图形软件开发方法初探——2D几何画板是一种利用Microsoft Visual C# 2.0作为开发语言,并基于Microsoft .NET Framework 2.0平台构建的2D图形软件。该软件旨在研究GDI+技术在二维图形软件开发中的应用...
基于大数据思维的银行监管数据应用初探--以3种数据挖掘技术为例.pdf
心理援助热线标准化管理流程建设初探--北京市心理援助热线电脑操作系统介绍.pdf
网络安全人才的多元主体协同育人初探--以中国网络空间安全人才教育联盟为例.pdf
中学地理计算机辅助教学初探-模板.pdf
医院消防安全管理初探-2页.pdf