前言

最近参加了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. binom_test

    bt <- function(a, b, p = 0.5) {binom.test(a, b+a, 0.5, alternative= c("two.sided"), con ...

  2. 转帖:弹性布局(display:flex;)属性详解

    它之所以被称为 Flexbox ,是因为它能够扩展和收缩 flex 容器内的元素,以最大限度地填充可用空间.与以前布局方式(如 table 布局和浮动元素内嵌块元素)相比,Flexbox 是一个更强大 ...

  3. Caused by: com.mysql.jdbc.exceptions.jdbc4.MySQLSyntaxErrorException: Table 'hive.DELETEME1643159643943' doesn't exist

    Caused by: com.mysql.jdbc.exceptions.jdbc4.MySQLSyntaxErrorException: Table 'hive.DELETEME1643159643 ...

  4. 利用 Rainbond 云原生平台简化 Kubernetes 业务问题排查

    Kubernetes 已经成为了云原生时代基础设施的事实标准,越来越多的应用系统在 Kubernetes 环境中运行.Kubernetes 已经依靠其强大的自动化运维能力解决了业务系统的大多数运行维护 ...

  5. TypeScript 学习总结

    TypeScript JavaScript 语言 面向对象编程语言 面向脚本编程 是否支持可选参数 支持 不支持 是否支持静态类型 支持 不支持 是否支持接口 支持 不支持 TS:是JS的超集,即对J ...

  6. noopener, noreferrer 及 nofollow 的用法

    <a> 标签通常会配合着使用 noopener, noreferrer 及 nofollow 这些属性, 它们的作用及用法如下. noopener 当给链接加上 target=" ...

  7. PyQt5学习 (1)--对象的基本操作、QObject

    参考视频:[Python-GUI编程-PyQt5 (少)] https://www.bilibili.com/video/BV17J41177ro/?share_source=copy_web& ...

  8. R语言文本数据挖掘(四)

    文本分词,就是对文本进行合理的分割,从而可以比较快捷地获取关键信息.例如,电商平台要想了解更多消费者的心声,就需要对消费者的文本评论数据进行内在信息的数据挖掘分析,而文本分词是文本挖掘的重要步骤.R语 ...

  9. [Python]Python安装教程

    anaconda Anaconda:python的一种软件发行版.Anaconda发行版会预装很多pydata生态圈里的软件,而Miniconda是最小的conda安装环境, 一个干净的conda环境 ...

  10. [Java EE]解决浏览器跨域问题

    1 解决浏览器跨域问题的方案 方式1: 浏览器(chrome)中取消跨域限制 step1 浏览器 chrome://flags step2 搜索:same step3 将搜索结果中的3个插件[Same ...