java arraylist底层实现原理_ArrayList和LinkedList底层原理

ArrayList和LinkedList都是List的实现类,是在日常开发中经常被使用到的两个集合,我们来结合源码看下两个集合的不同之处。

先来看下ArrayList的源码:

// 默认的初始化大小private static final int DEFAULT_CAPACITY = 10;

ArrayList的底层数数组结构,我们创建ArrayList的时候,可以使用指定数组大小的构造函数或者直接是默认的构造函数。当使用默认构造函数的时候,数组的初始化大小是0,当第一次调用add()方法的时候,会变成默认的初始化大小10。

使用ArrayList的最多的就是add()方法了,我们直接来看add()方法的源码:

public boolean add(E e) {//调用ensureCapacityInternal()看是否需要初始化以及扩容ensureCapacityInternal(size + 1); // Increments modCount!!//将数据添加到数组中elementData[size++] = e;return true;}

来看ensureCapacityInternal()方法:

private void ensureCapacityInternal(int minCapacity) {// 如果elementData为空,也就是第一次add的话,则返回默认容量和minCapacity中的最大值if (elementData == DEFAULTCAPACITY_EMPTY_ELEMENTDATA) {minCapacity = Math.max(DEFAULT_CAPACITY, minCapacity);}// ensureExplicitCapacity()方法则进一步判断是否需要扩容ensureExplicitCapacity(minCapacity);}private void ensureExplicitCapacity(int minCapacity) {modCount++;// 如果添加后的容量大于原来的容量,则调用扩容方法if (minCapacity - elementData.length > 0)grow(minCapacity);}private void grow(int minCapacity) {// 原来的容量int oldCapacity = elementData.length;//将原来的容量右移一位,相当于是*2^-1,总容量为原来的1.5倍int newCapacity = oldCapacity + (oldCapacity >> 1);if (newCapacity - minCapacity < 0)newCapacity = minCapacity;if (newCapacity - MAX_ARRAY_SIZE > 0)newCapacity = hugeCapacity(minCapacity);// 将旧数据拷贝到新数组中elementData = Arrays.copyOf(elementData, newCapacity);}

ArrayList的add方法逻辑很清晰,也没有过多的方法嵌套,就是添加数据到数组中,判断一下是否需要扩容,需要的话就进行扩容操作。

116c4a9b07fe822ba2f01ca5b11157cf.png山东掌趣网络科技

get方法:

public E get(int index) {rangeCheck(index);checkForComodification();return ArrayList.this.elementData(offset + index);}

get()就是直接获取该索引处的元素。

接下来我们就来分析下remove()方法:

remove(int index)方法:

public E remove(int index) {// 越界检查rangeCheck(index);modCount++;E oldValue = elementData(index);// 判断是否是最后一个元素int numMoved = size - index - 1;// 如果不是,则需要将index之后的元素往前移动一位if (numMoved > 0)System.arraycopy(elementData, index+1, elementData, index,numMoved);// 将最后一个元素删除,帮助GCelementData[--size] = null; // clear to let GC do its workreturn oldValue;}

remove(Object o)方法:

public boolean remove(Object o) {if (o == null) {for (int index = 0; index < size; index++)if (elementData[index] == null) {fastRemove(index);return true;}} else {for (int index = 0; index < size; index++)if (o.equals(elementData[index])) {fastRemove(index);return true;}}return false;}

remove逻辑比较简单,直接来看fastRemove()方法:

private void fastRemove(int index) {// modCount,继承于AbstractList。记录着集合的修改次数modCount++;int numMoved = size - index - 1;// 判断是否是最后一个元素,这里的操作和remove(index)一样if (numMoved > 0)System.arraycopy(elementData, index+1, elementData, index,numMoved);elementData[--size] = null; // clear to let GC do its work}

接下来我们来看下删除的例子:

37ccb9ff1c4aec56dc65d33ef3b3b7c4.png山东掌趣网络科技

public class ArrayListDemo {public static void main(String[] args) {List list=new ArrayList();list.add("jack");list.add("tom");list.add("tom");list.add("mary");list.add("danel");System.out.println("删除之前的集合--:"+list);// for循环删除字符串为tom的元素for(int i=0;i

我们看到,集合中元素为“tom”的并未完全删除掉,结合上面remove(int index)的源码,List调用remove(index)方法后,会移除index位置上的元素,index之后的元素就全部依次往前移,当删除掉一个元素后,size的值就变成了4,此时tom的索引位置为1,由于之前遍历的时候,i已经是1了,再次遍历的时候,i是从2开始,所以tom这个元素边不会再被遍历到,所以会存在漏删的情况。

e08918b170accef7be8843fca37d7a2e.png山东掌趣网络科技

再来看一个例子,使用foreach来遍历删除:

for(String name : list){if("tom".equals(name)){list.remove(name);}}Exception in thread "main" java.util.ConcurrentModificationExceptionat java.util.ArrayList$Itr.checkForComodification(ArrayList.java:901)at java.util.ArrayList$Itr.next(ArrayList.java:851)at ArrayListDemo.main(ArrayListDemo.java:14)

代码报了ConcurrentModificationException异常,为什么会出现这个异常呢,还记得上面源码分析的时候提到过一个成员变量modCount吗,modCount继承于AbstractList,记录着集合的修改次数,也就是每次add或者remove的话modCount值都会被修改,根据报错的地方,跟踪到它的源码:

final void checkForComodification() {if (modCount != expectedModCount)throw new ConcurrentModificationException();}

expectedModCount表示对ArrayList修改次数的期望值,我们来看下expectedModCount的源码位置,是内部类Itr的成员变量:

private class Itr implements Iterator {int cursor; // index of next element to returnint lastRet = -1; // index of last element returned; -1 if no suchint expectedModCount = modCount;public boolean hasNext() {return cursor != size;}、、、、

foreach之所以能工作,是因为这些集合类都实现了Iterable接口,该接口中定义了Iterator迭代器的产生方法,并且foreach就是通过Iterable接口在序列中进行移动,foreach语法最终被编译器转为了对Iterator.next()的调用,

每次foreach迭代的时候都有会有两步操作:

第一步:iterator.hasNext() //判断是否有下个元素

public boolean hasNext() {return cursor != size;}

第二步:下个元素是什么

public E next() {checkForComodification();int i = cursor;if (i >= size)throw new NoSuchElementException();Object[] elementData = ArrayList.this.elementData;if (i >= elementData.length)throw new ConcurrentModificationException();cursor = i + 1;return (E) elementData[lastRet = i];}

我们打印的异常也说明了这一点,Iterator初始化的时候将modCount 的值赋给了expectedModCount,此时这是一个固定值,而当调用remove方法的时候,会修改modCount ,当modCount 值与expectedModCount值不相等的时候,就会报ConcurrentModificationException异常。

那需要怎样删除ArrayList的数据呢?我们再看Iterator的源码,里面有一个删除的方法:

public void remove() {if (lastRet < 0)throw new IllegalStateException();checkForComodification();try {ArrayList.this.remove(lastRet);cursor = lastRet;lastRet = -1;expectedModCount = modCount;} catch (IndexOutOfBoundsException ex) {throw new ConcurrentModificationException();}}

这个remove方法的源码将修改后的modCount的值又重新复制给了expectedModCount ,所以不会再报ConcurrentModificationException异常,正确的删除是获取list集合的迭代器,然后通过迭代器调用它的remove方法。

Iterator iterator = list.iterator();while (iterator.hasNext()){String name = (String)iterator.next();if("tom".equals(name)){iterator.remove();}}

LinkedList源码分析:

相比于ArrayList,它额外实现了双端队列接口Deque,这个接口主要是声明了队头,队尾的一系列方法。LinkedList是一个双向链表结构,这里定义了链表的头结点和尾结点:

// 链表的头元素transient Node first;// 链表的尾元素transient Node last;

LinkedList的内部类,用以表示一个链表节点,一个节点除了保持自身的数据外,还持有前,后两个节点的引用。所以就数据存储上来说,它相比使用数组作为底层数据结构的ArrayList来说,会更加耗费空间。但也正因为这个特性,它删除,插入节点很快

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

每一个元素在LinkedList中都是在Node中存储,每个Node中同时还存储了当前元素的前节点和后节点。

我们再来看LikedList的add方法:

public boolean add(E e) {linkLast(e);return true;}

add方法没做什么事情,只调用了linkLast()方法,真正做事的是linkLast(E e),来看下这个方法:

void linkLast(E e) {// 将原尾结点赋值给lfinal Node l = last;// 新节点的头节点是原集合的尾结点,它自己的尾节点为空final Node newNode = new Node<>(l, e, null);// 将新节点赋值给尾节点last = newNode;// 如果原集合的尾结点是null的话,说明原集合没有值,它的头结点就是现 在的新增的数据if (l == null)first = newNode;else //否则的话将原尾结点的下一个节点指向当前新增的数据l.next = newNode;size++; // 集合数量+1modCount++; //修改次数+1}

add()方法是将数据插入到链表的尾部,通过add(int index, E element)方法可以在指定位置插入指定元素。

public void add(int index, E element) {// 检查指针是否越界checkPositionIndex(index);// 如果指定位置是最后一个的话,则直接在尾部插入if (index == size)linkLast(element);elselinkBefore(element, node(index));}

看下linkBefore(E e, Node succ)的源码:

void linkBefore(E e, Node succ) {// assert succ != null;final Node pred = succ.prev;// 新节点的前驱节点是原索引处节点的前驱节点,next节点是原索引处节点final Node newNode = new Node<>(pred, e, succ);// 将原索引处的节点的前驱节点修改为新的节点succ.prev = newNode;if (pred == null)first = newNode;else//将原前驱节点的next节点修改为新的节点pred.next = newNode;size++;modCount++;}

get方法源码:

Node node(int index) {// assert isElementIndex(index);// 首先判断该索引时位于前半段还是后半段if (index < (size >> 1)) {Node x = first;for (int i = 0; i < index; i++)x = x.next;return x;} else {Node x = last;for (int i = size - 1; i > index; i--)x = x.prev;return x;}}

LinkedList的get首先判断需要获取的数据在该链表的前半段还是后半段,这样可以减少需要遍历的数据,由于需要遍历数据,所以获取数据的速度会比ArrayList慢。

再来看下删除的方法

public E remove(int index) {// 检查下标是否越界checkElementIndex(index);return unlink(node(index));}E unlink(Node x) {// 得到要删除的元素final E element = x.item;// 获得删除元素的前驱和后置节点final Node next = x.next;final Node prev = x.prev;// 前驱节点为null的话说明要删除的节点是第一个节点if (prev == null) {first = next;} else {// 修改前驱节点的next节点指向被删除节点的next节点prev.next = next;x.prev = null;}// 后置节点为null的话说明要删除的节点是最后一个节点if (next == null) {last = prev;} else {// 修改next节点的前驱节点指向被删除节点的前驱节点next.prev = prev;x.next = null;}x.item = null;size--;modCount++;return element;}

remove(Object o)和remove(int index)稍有差别,但是基本都是调用unlink(Node x)方法。

在使用for循环或者是foreach迭代的时候,和ArrayList一样,也会有漏删或者报错,因此仍然建议使用迭代器中的remove方法进行迭代删除。

值得一提的是LinkedList还实现Dequeue接口,接口定义了队列数据结构,元素是有序的(按插入顺序),先进先出

可以利用LinkedList实现一个队列。LinkedList中的方法:

// 将指定的元素插入此队列public boolean offer(E e) {return add(e);}// 获取但不移除此队列的头;如果此队列为空,则返回 null。public E peek() {final Node f = first;return (f == null) ? null : f.item;}// 获取并移除此队列的头,如果此队列为空,则返回 null。public E poll() {final Node f = first;return (f == null) ? null : unlinkFirst(f);}// 获取并移除此队列的头public E remove() {return removeFirst();}、、、

例如简单的队列:

public class DequeDemo {public static void main(String[] args) {Queue queue = new LinkedList();queue.offer("tom");queue.offer("jack");queue.offer("mike");while (queue.peek()!=null){System.out.println(queue.poll());}}}

作为双端队列的一些方法,如: addFirst(Object ob):在队首增加元素addLast(Object obj):在队尾增加;peekFirst():查看队首;peekLast:查看队尾;pollFirst:移除队首;pollLast:移除队尾例如:

public class DequeDemo {public static void main(String[] args) {Deque queue = new LinkedList();queue.offer("tom");queue.offer("jack");queue.offer("mike");queue.addFirst("dannel");queue.addLast("mary");while (queue.peek()!=null){System.out.println(queue.poll());}}}danneltomjackmikemary

由于LinkedList底层是用双向链表实现的,没有初始化大小,也没有扩容的机制。

总结

ArrayList和LinkedList都实现了List的接口,但是LinkedList还实现了Deque队列接口,因此还可以当做队列使用,

ArrayList的底层结构是数组,获取元素的速度快,但是插入速度相对LinkedList较慢,LinkedList的底层是双向链表结构,插入和删除比较快,但是查找的速度相对ArrayList较慢。另外ArrayList和LinkedList在并发情况下没有对数据进行同步,是线程不安全的。

山东掌趣网络科技

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/533361.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

java ee jaas_java-ee – Tomcat-Jaas – 如何检索主题?

i knew that and it works, but I need to retrieve subject to get also roleprincipal不幸的是,它在Java EE中的工作方式不同. JAAS主题只是一个“主要包”,其中哪些代表用户/调用者主体和/或角色主体根本不是标准化的.每个其他容器在这里做不同的事情. Javadoc for Tomcat’…

java jive歌词_Java Jive_Manhattan Transfer with Phil Collins_高音质在线试听_Java Jive歌词|歌曲下载_酷狗音乐...

Manhattan Transfer with Phil Collins - Java Jive&#xfeff;[id:$00000000][ar:曼哈顿行者爵士][ti:Java Jive (LP Version)][by:][hash:99bf26cac4ad13e15925a56eb724027f][al:][sign:][qq:][total:0][offset:0][00:00.05]The Manhattan Transfer - Java Jive[00:10.57]I …

java 3_Java 3 (Java的数据类型)

Java的数据类型主要内容&#xff1a;1Java数据类型的分类2.8种基本数据类型3.理解引用类型的特点一、什么是数据类型&#xff1f;计算机语言将数据按性质进行分类&#xff0c;每一类称为一种数据类型&#xff1b;数据类型定义了数据的性质、取值范围、存储方式、对数据所能进行…

java快捷键 --_Java中的快捷方式“或分配”(| =)运算符

如果是关于可读性&#xff0c;我就有了将测试数据与测试逻辑分离的概念。代码示例&#xff1a;// declare dataDataType [] dataToTest new DataType[] {defaultStock,defaultWholesale,defaultRetail,defaultDelivery}// define logicboolean checkIfAnyNegative(DataType []…

tcp网络通信教程 java_基于java TCP网络通信的实例详解

JAVA中设计网络编程模式的主要有TCP和UDP两种&#xff0c;TCP是属于即时通信&#xff0c;UDP是通过数据包来进行通信&#xff0c;UDP当中就会牵扯到数据的解析和传送。在安全性能方面&#xff0c;TCP要略胜一筹&#xff0c;通信过程中不容易出现数据丢失的现象&#xff0c;有一…

java博客论坛设计报告_javaweb课程设计报告个人博客网站的实现(Java).doc

javaweb课程设计报告个人博客网站的实现(Java)项目名称&#xff1a; 个人博客网站的实现(Java) 学生姓名&#xff1a;学 号&#xff1a;班 级&#xff1a;指导教师&#xff1a;2014年12月23日目录1 绪论11.1系统应用意义11.2主要设计任务11.3开发及运行环境11.3.1 JSP的基础——…

java replace stringbuilder_java.lang.StringBuilder.replace()方法实例

全屏java.lang.StringBuilder.replace()方法按照这个顺序&#xff0c;在指定的字符串的子字符串替换字符。子串开始在指定start的 索引&#xff0c;并延伸到该字符 end - 1&#xff0c;或如果序列的末端不存在这样的字符。声明以下是java.lang.StringBuilder.replace()方法的声…

中小学课java_java毕业设计_springboot框架的中小学排课与实现

这是一个基于java的毕业设计项目,毕设课题为springboot框架的中小学排课与实现, 是一个采用b/s结构的javaweb项目, 开发工具eclipsei/eclipse, 项目框架jspspringbootmybatis, 中小学排课与实现采用mysql进行数据存储, 并基于mybatis进行了orm实体关系映射, 该中小学排课与实现…

java 文件设置为只读文件系统_Java如何设置文件为只读?

在java编程中&#xff0c;如何设置文件为只读&#xff1f;此示例演示如何使用File类的file.setReadOnly()和file.canWrite()方法设置文件为只读模式。package com.yiibai;import java.io.File;public class ReadOnlyFile {public static void main(String[] args) {File file …

wordcount linux java_linux下在eclipse上运行hadoop自带例子wordcount

启动eclipse&#xff1a;打开windows->open perspective->other->map/reduce 可以看到map/reduce开发视图。设置Hadoop location.打开windows->show view->other-> map/reduce Locations视图&#xff0c;在点击大象后【new Hadoop location】弹出的对话框(Ge…

php java执行linux_java_java执行Linux命令的方法,本文实例讲述了java执行Linux命 - phpStudy...

java执行Linux命令的方法本文实例讲述了java执行Linux命令的方法。分享给大家供大家参考。具体实现方法如下&#xff1a;public class StreamGobbler extends Thread {InputStream is;String type;public StreamGobbler(InputStream is, String type) {this.is is;this.type …

java怎么接收前端请求_前端json post 请求 后端怎么接收

前端提交POST /api/test HTTP/1.1Host: 192.168.135.69:81Connection: keep-aliveContent-Length: 18Origin: http://192.168.135.69:81User-Agent: Mozilla/5.0 (iPhone; CPU iPhone OS 11_0 like Mac OS X) AppleWebKit/604.1.38 (KHTML, like Gecko) Version/11.0 Mobile/15…

minimum在java中的意思_Java Calendar getMinimum()用法及代码示例

Calendar类中的getMinimum(int calndr_field)方法用于返回此Calendar实例的给定日历字段(int calndr_field)的最小值。用法:public abstract int getMinimum(int calndr_field)参数&#xff1a;该方法采用一个参数calndr_field&#xff0c;该参数表示要操作的日历字段。返回值&…

django mysql 一对多_请教,django中 如何向带有外键(一对多和多对多)数据库中批量插入数据?...

已自行解决&#xff0c;代码如下&#xff1a;json格式&#xff1a;[{"标题": "小武","内容": "测试","类型":["情感","文学","散文"]"文章资源":[{"title":"小武.1…

安装php no permision,php安装过程中的No package ‘xxx’ found问题

php No package ‘oniguruma’ found今天安装php7.4的时候遇到这样的一个报错&#xff0c;然后yum install oniguruma oniguruma-devel&#xff0c;重试安装php&#xff0c;依然报错&#xff0c;又编译安装oniguruma&#xff0c;重试安装php&#xff0c;还是报错&#xff0c;问…

php httpclient.class.php,php实现httpclient类示例

class httpClient {public $buffer null; // buffer 获取返回的字符串public $referer null; // referer 设置 HTTP_REFERER 的网址public $response null; // response 服务器响应的 header 信息public $request null; // request 发送到服务器的 header 信息private $…

大学php老师,php高校教师总结计划系统

通过使用本系统&#xff0c;可以规范工作流程&#xff0c;提高办公效率&#xff0c;增强团队协同工作能力&#xff0c;实现科学的公文处理、事物管理、会议安排和人力管理&#xff0c;量化运营资源&#xff0c;预防管理真空&#xff0c;降低运行成本。还可以实现便利的信息发布…

好用的php空间,推荐国内三个优质的免费PHP空间

1.亿家免费国内PHP空间这是我见过最好的免费国内PHP空间了&#xff0c;这个BLOG就是由他的空间支撑的&#xff0c;所以你看到我这个空间的稳定&#xff0c;快速就代表着他们空间的优质了&#xff0c;推荐注册地址&#xff1a;www.e9china.net这个先要在他们论坛上发帖子&#x…

java处理脏数据,Java程序的脏数据问题

脏数据(Out-of-date data)&#xff0c;指过时的数据。假如在您的java程序中存在脏数据&#xff0c;将或多或少地给软件系统带来一些问题&#xff0c;如&#xff1a;无法实时地应用已经发生改变的配置&#xff0c;软件系统出现一些莫名其妙的、难以重现的、后果严重的错误等等。…

制作自己的 Docker 容器

软件开发最大的麻烦事之一&#xff0c;就是环境配置。用户必须保证操作系统的设置&#xff0c;各种库和组件的安装&#xff0c;只有它们都正确&#xff0c;软件才能运行。docker从根本上解决问题&#xff0c;软件安装的时候&#xff0c;把原始环境一模一样地复制过来。 以 koa-…