Java Collection - ArrayList LinkedList

2024-06-08 08:38

本文主要是介绍Java Collection - ArrayList LinkedList,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

ArrayList

基础属性:

transient Object[] elementData; 存储数据的数组

private static final int DEFAULT_CAPACITY = 10; 默认长度

private static final Object[] DEFAULTCAPACITY_EMPTY_ELEMENTDATA = {}; default sized empty instances

1. new ArrayList<>()

/*** Constructs an empty list with an initial capacity of ten.默认的数组长度是10*/
public ArrayList() {this.elementData = DEFAULTCAPACITY_EMPTY_ELEMENTDATA;
}

2. add方法 - 增

    public boolean add(E e) {ensureCapacityInternal(size + 1);  // Increments modCount!!//放在数组的最后elementData[size++] = e;return true;}private void ensureCapacityInternal(int minCapacity) {ensureExplicitCapacity(calculateCapacity(elementData, minCapacity));}private void ensureExplicitCapacity(int minCapacity) {//并发标识位modCount++;// overflow-conscious codeif (minCapacity - elementData.length > 0)//扩容!!grow(minCapacity);}private void grow(int minCapacity) {// overflow-conscious codeint oldCapacity = elementData.length;//扩容为原始长度的1.5倍。 >>右移位符号,移动一位相当于除以2。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);}//使用Arrays工具类的copyOf方法,底层使用的是System.arraycopy的native方法。可能会造成Heap space OOM

3. remove方法 - 删(有多个方法)

     public E remove(int index) {//检查下标,防止越界rangeCheck(index);modCount++;E oldValue = elementData(index);int numMoved = size - index - 1;if (numMoved > 0)//index之后的数据全部复制到index的位置。这一步较为费时,频繁的删除时不应该使用ArrayList!System.arraycopy(elementData, index+1, elementData, index,numMoved);//将数组指向null,方便GCelementData[--size] = null; // clear to let GC do its workreturn oldValue;}

4. set方法 - 改

public E set(int index, E element) {//检查下标后直接更新rangeCheck(index);E oldValue = elementData(index);elementData[index] = element;return oldValue;
}

5. get方法 -查

	//1、知道下标时  public E get(int index) {//大于等于size时,抛出IndexOutOfBoundsExceptionrangeCheck(index);//返回对应下标下的数据return elementData(index);}

LinkedList

基础属性:

transient Node first; 头结点

transient Node last; 尾节点

Node是一个内部类,定义如下:

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

1. new LinkedList<>();

/*** Constructs an empty list.*/
public LinkedList() {
}

2. 增 - add

    public boolean add(E e) {linkLast(e);return true;}//采用尾插法,将插入的节点作为last节点void linkLast(E e) {final Node<E> l = last;final Node<E> newNode = new Node<>(l, e, null);last = newNode;if (l == null)first = newNode;elsel.next = newNode;size++;modCount++;}

3. 删 - remove

   //remove指定index的public E remove(int index) {//检查下标,越界则抛出异常checkElementIndex(index);return unlink(node(index));}//获取指定index下的NodeNode<E> node(int index) {// assert isElementIndex(index);//可以看到此处作了优化,根据index在size的前半段或后半段,分别从前往后或从后往前查找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;}}//释放指定的NodeE unlink(Node<E> x) {// assert x != null;final E element = x.item;final Node<E> next = x.next;    //Node的前置节点nextfinal Node<E> prev = x.prev;    //Node的后置节点prev//Node是初始节点,将next置为first节点if (prev == null) {first = next;} else {//Node不是初始节点prev.next = next;x.prev = null;}//Node是尾节点,将前置节点作为尾节点if (next == null) {last = prev;} else {//Node不是尾节点,设置其前置节点next.prev = prev;x.next = null;}//Node变为了null,方便GC回收x.item = null;size--;modCount++;return element;}

可以看到删除时仅仅移动了链表,不像ArrayList一样对数组进行了Copy。

4. 改 - set

改特定位置的Node的数据

public E set(int index, E element) {//下面这两个方法很眼熟吧 :)checkElementIndex(index);Node<E> x = node(index);E oldVal = x.item;x.item = element;return oldVal;
}

5. 查 - get

public E get(int index) {checkElementIndex(index);return node(index).item;
}
  • 查找时ArrayList比LinkedList效率高,因为数组结构的遍历只需将下标加一,而链表结构则从头查找;
  • 对元素进行增删时LinedList会比ArrayList好用,链表结构可以很轻松的在链表中间去增删元素,而不需要改变插入节点后面元素的任何数据。

遍历

  • for循环遍历,实际上是使用get(int index)方法进行查找元素,最基础的

  • 迭代器,ArrayList和LinkedList均有内部类实现了ListIterator接口,如LinkedList:

      private class ListItr implements ListIterator<E> {private Node<E> lastReturned;private Node<E> next;private int nextIndex;private int expectedModCount = modCount;...省略部分代码public boolean hasNext() {return nextIndex < size;}public E next() {checkForComodification();if (!hasNext())throw new NoSuchElementException();lastReturned = next;next = next.next;nextIndex++;return lastReturned.item;}...省略public void remove() {checkForComodification();if (lastReturned == null)throw new IllegalStateException();Node<E> lastNext = lastReturned.next;unlink(lastReturned);if (next == lastReturned)next = lastNext;elsenextIndex--;lastReturned = null;expectedModCount++;}public void set(E e) {if (lastReturned == null)throw new IllegalStateException();checkForComodification();lastReturned.item = e;}public void add(E e) {checkForComodification();lastReturned = null;if (next == null)linkLast(e);elselinkBefore(e, next);nextIndex++;expectedModCount++;}
    

如果想一边迭代一边用集合自带的方法进行删除或者新增操作,都会抛出ConcurrentModificationException异常(在增加和删除元素的时候,都会进行自增操作 modCount)。所以在遍历时需要增删list元素的,则需要使用迭代器。

  • forEach,ArrayList使用的是数组下标:
    @Overridepublic void forEach(Consumer<? super E> action) {Objects.requireNonNull(action);final int expectedModCount = modCount;@SuppressWarnings("unchecked")final E[] elementData = (E[]) this.elementData;final int size = this.size;for (int i=0; modCount == expectedModCount && i < size; i++) {action.accept(elementData[i]);}if (modCount != expectedModCount) {throw new ConcurrentModificationException();}}

Best Practice:

  • 实现了RadmoAcces接口的list(如ArrayList),优先选择普通for循环(使用下标直接随机读取) ,其次list.foreach方法或增强型的for循环
  • 未实现RadmoAcces接口的list(如LinkedList), 优先选择iterator遍历,移动指针(foreach遍历底层也是通过iterator实现的),大size的数据,千万不要使用普通for循环(每次查找都要移动大量数据)

这篇关于Java Collection - ArrayList LinkedList的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1041691

相关文章

Java五子棋之坐标校正

上篇针对了Java项目中的解构思维,在这篇内容中我们不妨从整体项目中拆解拿出一个非常重要的五子棋逻辑实现:坐标校正,我们如何使漫无目的鼠标点击变得有序化和可控化呢? 目录 一、从鼠标监听到获取坐标 1.MouseListener和MouseAdapter 2.mousePressed方法 二、坐标校正的具体实现方法 1.关于fillOval方法 2.坐标获取 3.坐标转换 4.坐

Spring Cloud:构建分布式系统的利器

引言 在当今的云计算和微服务架构时代,构建高效、可靠的分布式系统成为软件开发的重要任务。Spring Cloud 提供了一套完整的解决方案,帮助开发者快速构建分布式系统中的一些常见模式(例如配置管理、服务发现、断路器等)。本文将探讨 Spring Cloud 的定义、核心组件、应用场景以及未来的发展趋势。 什么是 Spring Cloud Spring Cloud 是一个基于 Spring

Javascript高级程序设计(第四版)--学习记录之变量、内存

原始值与引用值 原始值:简单的数据即基础数据类型,按值访问。 引用值:由多个值构成的对象即复杂数据类型,按引用访问。 动态属性 对于引用值而言,可以随时添加、修改和删除其属性和方法。 let person = new Object();person.name = 'Jason';person.age = 42;console.log(person.name,person.age);//'J

java8的新特性之一(Java Lambda表达式)

1:Java8的新特性 Lambda 表达式: 允许以更简洁的方式表示匿名函数(或称为闭包)。可以将Lambda表达式作为参数传递给方法或赋值给函数式接口类型的变量。 Stream API: 提供了一种处理集合数据的流式处理方式,支持函数式编程风格。 允许以声明性方式处理数据集合(如List、Set等)。提供了一系列操作,如map、filter、reduce等,以支持复杂的查询和转

Java面试八股之怎么通过Java程序判断JVM是32位还是64位

怎么通过Java程序判断JVM是32位还是64位 可以通过Java程序内部检查系统属性来判断当前运行的JVM是32位还是64位。以下是一个简单的方法: public class JvmBitCheck {public static void main(String[] args) {String arch = System.getProperty("os.arch");String dataM

详细分析Springmvc中的@ModelAttribute基本知识(附Demo)

目录 前言1. 注解用法1.1 方法参数1.2 方法1.3 类 2. 注解场景2.1 表单参数2.2 AJAX请求2.3 文件上传 3. 实战4. 总结 前言 将请求参数绑定到模型对象上,或者在请求处理之前添加模型属性 可以在方法参数、方法或者类上使用 一般适用这几种场景: 表单处理:通过 @ModelAttribute 将表单数据绑定到模型对象上预处理逻辑:在请求处理之前

eclipse运行springboot项目,找不到主类

解决办法尝试了很多种,下载sts压缩包行不通。最后解决办法如图: help--->Eclipse Marketplace--->Popular--->找到Spring Tools 3---->Installed。

JAVA读取MongoDB中的二进制图片并显示在页面上

1:Jsp页面: <td><img src="${ctx}/mongoImg/show"></td> 2:xml配置: <?xml version="1.0" encoding="UTF-8"?><beans xmlns="http://www.springframework.org/schema/beans"xmlns:xsi="http://www.w3.org/2001

Java面试题:通过实例说明内连接、左外连接和右外连接的区别

在 SQL 中,连接(JOIN)用于在多个表之间组合行。最常用的连接类型是内连接(INNER JOIN)、左外连接(LEFT OUTER JOIN)和右外连接(RIGHT OUTER JOIN)。它们的主要区别在于它们如何处理表之间的匹配和不匹配行。下面是每种连接的详细说明和示例。 表示例 假设有两个表:Customers 和 Orders。 Customers CustomerIDCus

22.手绘Spring DI运行时序图

1.依赖注入发生的时间 当Spring loC容器完成了 Bean定义资源的定位、载入和解析注册以后,loC容器中已经管理类Bean 定义的相关数据,但是此时loC容器还没有对所管理的Bean进行依赖注入,依赖注入在以下两种情况 发生: 、用户第一次调用getBean()方法时,loC容器触发依赖注入。 、当用户在配置文件中将<bean>元素配置了 lazy-init二false属性,即让