前言

最近参加了21天打卡活动,希望可以让自己养成写博客的习惯…

ArrayList和LinkedList

ArrayList和LinkedList都是常用的List类型,两者都继承了AbstratctList,并实现List接口。

List的方法

列举一些常见的方法,ArrayList和LinkedList会实现List里面的方法

方法 描述
boolean isEmpty() 判断当前列表是否为空
boolean contains(Object o) 是否包含这个元素
T[] toArray(T[] a)
boolean add(E e) 添加一个元素
boolean remove(Object o) 移除
boolean containsAll(Collection<?> c)
void add(int index, E element) 在固定位置添加元素
int indexOf(Object o) 定位元素
ListIterator listIterator() 遍历列表

上述的方法都是List的常用方法,相信大家都非常的熟悉。

ArrayList

ArrayList是实现List接口的可扩容数组(动态数组),它的内部是基于数组实现的,数组这个结构具有的特点

随机存取:随机存取就是可以通过索引直接访问列表的元素

可以实现动态扩容,下面我们来看一下它的源码实现

属性 描述
size 列表的元素个数
DEFAULT_CAPACITY 创建一个空对象时的默认大小

我们来看一下ArrayList是怎么实现动态扩容的。

 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);
}

如果当容量比初始容量大时,新的容量相当于是原来的1.5倍这里

add

ArrayList重载了List的add方法,这里面重写的方法有直接添加,和在某一个位置添加一个元素,这里就可以看出ArrayList的底层是由数组实现的了

public boolean add(E e) {
ensureCapacityInternal(size + 1); // Increments modCount!!
elementData[size++] = e;
return true;
} public void add(int index, E element) {
rangeCheckForAdd(index); ensureCapacityInternal(size + 1); // Increments modCount!!
System.arraycopy(elementData, index, elementData, index + 1,
size - index);
elementData[index] = element;
size++;
}

这里采用的是System.arraycopy来进行数组的拷贝,属于深度拷贝,效率很高。

remove

这里的移除也是有直接移除某个索引下的数据,但是这里面有两个不同的方式移除数据,一个是remove,一个是fastRemove两种方式移除元素,remove和fastRemove不同的是一个是需要返回删除的元素,一个是不返回的。

LinkedList

LinkedList是基于双向链表来实现的,是属于链式存储,只能顺序存取元素,不能随机存取。

LinkedList的结点结构,包括指向下一个结点的指针next和指向上一个结点的指针prev

private static class Node<E> {
E item;
Node<E> next;
Node<E> prev; Node(Node<E> prev, E element, Node<E> next) {
this.item = element;
this.next = next;
this.prev = prev;
}
}

属性包括

名称 描述
size 元素个数
first 头节点
last 尾节点

remove

移除头节点,需要更新size大小和把删除的节点为NULL,里面还有removeRange方法,可以把从fromIndex到toIndex的全部节点释放掉

 public E remove() {
return removeFirst();
}
private E unlinkLast(Node<E> l) {
// assert l == last && l != null;
final E element = l.item;
final Node<E> prev = l.prev;
l.item = null;
l.prev = null; // help GC
last = prev;
if (prev == null)
first = null;
else
prev.next = null;
size--;
modCount++;
return element;
} public E removeLast() {
final Node<E> l = last;
if (l == null)
throw new NoSuchElementException();
return unlinkLast(l);
}

get和peek

下面可以来看看LinkedList的访问元素的形式,前面说了这个结构是只能顺序遍历的,不能随机进行访问,需要遍历整个列表,但是我们使用的是双向链表,由于我们维护了头节点和尾节点,当需要访问元素时,如果根据的是下标(这里的下标不是数组的下标),先判断和头节点近还是和尾节点近,然后再进行顺序遍历。

peek方法是访问头节点。
Node<E> node(int index) {
// assert isElementIndex(index);
// 往后
if (index < (size >> 1)) {
Node<E> x = first;
for (int i = 0; i < index; i++)
x = x.next;
return x;
} else {
Node<E> x = last;
//往前
for (int i = size - 1; i > index; i--)
x = x.prev;
return x;
}
}

push

向列表中添加元素,我们可以采用offerFirst和offerLast往列表头部添加元素或者往尾部添加元素,push方法调用的是addFrist方法,采用的是头插法,删除也是才有删除头部元素;

ArrayList和LinkedList的使用场景和区别

前面也提到两者的存取结构是不同的,一个是用数组来操作,一个采用的是双向链表,一般ArrayList用在访问更加多的情况下,由于插入会相对较慢,但是LinkedList采用的是顺序访问,在插入和删除较多的场景会更加适用。

浅谈ArrayList和LinkedList的更多相关文章

  1. 浅谈ArrayList

    浅谈ArrayList 废话不多说(事实是不会说),让我们直接进入正题 首先讲一讲最基本的ArrayList的初始化,也就是我们常说的构造函数,ArrayList给我们提供了三种构造方式,我们逐个来查 ...

  2. 浅谈 ArrayList 及其扩容机制

    浅谈ArrayList ArrayList类又称动态数组,同时实现了Collection和List接口,其内部数据结构由数组实现,因此可对容器内元素实现快速随机访问.但因为ArrayList中插入或删 ...

  3. 浅谈Java中linkedlist和arraylist区别

    在Java中,关于集合框架有这样一个体系结构: 其主要由两个接口派生而出:Collection和Map,然后再衍生出各自的一些实现类(比如Collection接口又被继承与Set和List接口,而他们 ...

  4. 简谈ArrayList和LinkedList区别

    对于ArrayList和LinkedList,他们都实现了List接口,他们的区别大致为: ArrayList LinkedList (1)底层是数组,可以以O(1)的时间复杂度对元素进行随机访问 以 ...

  5. 浅谈 Java集合

    Java 集合 集合是对象的容器,定义了多个对象进行操作的常用方法,可实现数组的功能. Java集合类库所处位置:java.util.*. 与现代的数据结构类库的常见做法一样,Java集合类库也将接口 ...

  6. 浅谈Vector、ArrayList、LinkedList

    下图是Collection的类继承图 从图中可以看出:Vector.ArrayList.LinkedList这三者都实现了List 接口.所有使用方式也很相似,主要区别在于实现方式的不同,所以对不同的 ...

  7. 浅谈JAVA集合框架

    浅谈JAVA集合框架 Java提供了数种持有对象的方式,包括语言内置的Array,还有就是utilities中提供的容器类(container classes),又称群集类(collection cl ...

  8. 浅谈Java的集合框架

    浅谈Java的集合框架 一.    初识集合 重所周知,Java有四大集合框架群,Set.List.Queue和Map.四种集合的关注点不同,Set 关注事物的唯一性,List 关注事物的索引列表,Q ...

  9. 浅谈java类集框架和数据结构(2)

    继续上一篇浅谈java类集框架和数据结构(1)的内容 上一篇博文简介了java类集框架几大常见集合框架,这一篇博文主要分析一些接口特性以及性能优化. 一:List接口 List是最常见的数据结构了,主 ...

  10. 浅谈Java中set.map.List的区别

    就学习经验,浅谈Java中的Set,List,Map的区别,对JAVA的集合的理解是想对于数组: 数组是大小固定的,并且同一个数组只能存放类型一样的数据(基本类型/引用类型),JAVA集合可以存储和操 ...

随机推荐

  1. jmeter学习-性能指标、jmeter初识

    一:性能测试的指标 1. 并发/并发数/并发用户数 狭义的并发:同一时间做相同的一件事 广义的并发:同一时间做不同事情,混合场景,对服务器来说的并发 性能测试,先做简单的狭义并发,在做广义并发:先做单 ...

  2. MFS分布式存储特性及组件说明

    1.MFS MooseFS是一个具有冗余容错功能的分布式网络文件系统,它将数据分别存放在多个物理服务器或单独磁盘或分区上,确保一份数据有多个备份副本,然而对于访问MFS的客户端或者用户来说,整个分布式 ...

  3. Go_day04

    Go基础语法 指针 指针式存储另一个变量内存地址的变量 &a 取出a的内存地址 *b 若指针b存放的式a的地址 那么 *b就直接指向a的内存 可以直接操作其中的值 指针的使用 func mai ...

  4. sqlserver 生成随机值

    随机生成100以内的整数 select top 10 number,ceiling(rand(checksum(newid()))*100) from master.dbo.spt_valueswhe ...

  5. 07-Spring的事务处理

    Spring中提供了七种事务的传播行为: PROPAGATION_REQUIRED:默认值,最常用,统一事务,出现异常,全部回滚 其余参考Spring事务处理word文档. 0.原转账业务(不含事务处 ...

  6. DVWA-XSS (Stored) 存储型XSS

    存储型XSS,顾名思义,就是会传入数据库,长久的使用,常见为留言板,用户信息资料. LOW 审计源码 <?php // 是否提交 btnSign if( isset( $_POST[ 'btnS ...

  7. 使用 ApplicationContextAware 定义 SpringContextHolder 类

    需求:使用 @autowired注入一些对象,但发现不可以直接使用@Autowired,因为方法是static的,要使用该方法当前对象也必须是static,正常情况下@Autowired无法注入静态的 ...

  8. Java 遍历方式

    一.遍历方式 迭代器 增强for循环 普通for循环 二.使用 迭代器: public class IteratorMethod { public static void main(String[] ...

  9. VS2013连接SQLSERVER数据库时显示无法添加数据连接

    VS2013连接sqlserver2008时出现如下错误: VS2013添加Microsoft SQL Server数据源时,会遇到无法添加数据连接. Could not load file or a ...

  10. Mac 音频转换器推荐 DRmare Audio Converter、Audi Free Auditor

    给大家推荐两款 Mac 上的音频转换器,这两款转换器都可以转换苹果音乐,iTunes歌曲或者一些常规的音轨到MP3, FLAC, WAV, M4A, AAC格式等等,转换后我们就可以在所有的设备和播放 ...