iis服务器助手广告广告
返回顶部
首页 > 资讯 > 后端开发 > JAVA >java中什么情况下使用ArrayList和LinkedList?
  • 885
分享到

java中什么情况下使用ArrayList和LinkedList?

java教程ArrayListLinkedList 2016-08-06 11:08:07 885人浏览 猪猪侠
摘要

什么情况用ArrayList or LinkedList呢?ArrayList 和 LinkedList 是 Java 集合框架中用来存储对象引用列表的两个类。ArrayList 和 LinkedList 都实现 List 接口。先对Lis

什么情况用ArrayList or LinkedList呢?

ArrayList 和 LinkedList 是 Java 集合框架中用来存储对象引用列表的两个类。ArrayList 和 LinkedList 都实现 List 接口。先对List做一个简单的了解:

列表(list)是元素的有序集合,也称为序列。它提供了基于元素位置的操作,有助于快速访问、添加和删除列表中特定索引位置的元素。List 接口实现了 Collection 和 Iterable 作为父接口。它允许存储重复值和空值,支持通过索引访问元素。

读完这篇文章要搞清楚的问题:ArrayList和LinkedList有什么不同之处?什么时候应该用ArrayList什么时候又该用LinkedList呢?

(推荐视频:java视频教程

下面以增加和删除元素为例比较ArrayList和LinkedList的不同之处

增加元素到列表尾端:

在ArrayList中增加元素到队列尾端的代码如下:

public boolean add(E e){
   ensureCapacity(size+1);//确保内部数组有足够的空间
   elementData[size++]=e;//将元素加入到数组的末尾,完成添加
   return true;      
}

ArrayList中add()方法的性能决定于ensureCapacity()方法。ensureCapacity()的实现如下:

public vod ensureCapacity(int minCapacity){
  modCount++;
  int oldCapacity=elementData.length;
  if(minCapacity>oldCapacity){    //如果数组容量不足,进行扩容
      Object[] oldData=elementData;
      int newCapacity=(oldCapacity*3)/2+1;  //扩容到原始容量的1.5倍
      if(newCapacitty

可以看到,只要ArrayList的当前容量足够大,add()操作的效率非常高的。只有当ArrayList对容量的需求超出当前数组大小时,才需要进行扩容。扩容的过程中,会进行大量的数组复制操作。而数组复制时,最终将调用System.arraycopy()方法,因此add()操作的效率还是相当高的。

LinkedList 的add()操作实现如下,它也将任意元素增加到队列的尾端:

public boolean add(E e){
   addBefore(e,header);//将元素增加到header的前面
   return true;
}

其中addBefore()的方法实现如下:

private Entry addBefore(E e,Entry entry){
     Entry newEntry = new Entry(e,entry,entry.previous);
     newEntry.provious.next=newEntry;
     newEntry.next.previous=newEntry;
     size++;
     modCount++;
     return newEntry;
}

可见,LinkeList由于使用了链表的结构,因此不需要维护容量的大小。从这点上说,它比ArrayList有一定的性能优势,然而,每次的元素增加都需要新建一个Entry对象,并进行更多的赋值操作。在频繁的系统调用中,对性能会产生一定的影响。

增加元素到列表任意位置

除了提供元素到List的尾端,List接口还提供了在任意位置插入元素的方法:void add(int index,E element);

由于实现的不同,ArrayList和LinkedList在这个方法上存在一定的性能差异,由于ArrayList是基于数组实现的,而数组是一块连续的内存空间,如果在数组的任意位置插入元素,必然导致在该位置后的所有元素需要重新排列,因此,其效率相对会比较低。

以下代码是ArrayList中的实现:

public void add(int index,E element){
   if(index>size||index<0)
      throw new IndexOutOfBoundsException(
        "Index:"+index+",size: "+size);
         ensureCapacity(size+1);
         System.arraycopy(elementData,index,elementData,index+1,size-index);
         elementData[index] = element;
         size++;
}

可以看到每次插入操作,都会进行一次数组复制。而这个操作在增加元素到List尾端的时候是不存在的,大量的数组重组操作会导致系统性能低下。并且插入元素在List中的位置越是靠前,数组重组的开销也越大。

而LinkedList此时显示了优势:

public void add(int index,E element){
   addBefore(element,(index==size?header:entry(index)));
}

可见,对LinkedList来说,在List的尾端插入数据与在任意位置插入数据是一样的,不会因为插入的位置靠前而导致插入的方法性能降低。

删除任意位置元素

对于元素的删除,List接口提供了在任意位置删除元素的方法:

public E remove(int index);

对ArrayList来说,remove()方法和add()方法是雷同的。在任意位置移除元素后,都要进行数组的重组。ArrayList的实现如下:

public E remove(int index){
   RangeCheck(index);
   modCount++;
   E oldValue=(E) elementData[index];
  int numMoved=size-index-1;
  if(numMoved>0)
     System.arraycopy(elementData,index+1,elementData,index,numMoved);
     elementData[--size]=null;
     return oldValue;
}

可以看到,在ArrayList的每一次有效的元素删除操作后,都要进行数组的重组。并且删除的位置越靠前,数组重组时的开销越大。

public E remove(int index){
  return remove(entry(index));         
}
private Entry entry(int index){
  if(index<0 || index>=size)
      throw new IndexOutBoundsException("Index:"+index+",size:"+size);
      Entry e= header;
      if(index<(size>>1)){//要删除的元素位于前半段
         for(int i=0;i<=index;i++)
             e=e.next;
     }else{
         for(int i=size;i>index;i--)
             e=e.previous;
     }
         return e;
}

在LinkedList的实现中,首先要通过循环找到要删除的元素。如果要删除的位置处于List的前半段,则从前往后找;若其位置处于后半段,则从后往前找。因此无论要删除较为靠前或者靠后的元素都是非常高效的;但要移除List中间的元素却几乎要遍历完半个List,在List拥有大量元素的情况下,效率很低。

容量参数

容量参数是ArrayList和Vector等基于数组的List的特有性能参数。它表示初始化的数组大小。当ArrayList所存储的元素数量超过其已有大小时。它便会进行扩容,数组的扩容会导致整个数组进行一次内存复制。因此合理的数组大小有助于减少数组扩容的次数,从而提高系统性能。

public  ArrayList(){
  this(10);  
}
public ArrayList (int initialCapacity){
   super();
   if(initialCapacity<0)
       throw new IllegalArgumentException("Illegal Capacity:"+initialCapacity)
      this.elementData=new Object[initialCapacity];
}

ArrayList提供了一个可以制定初始数组大小的构造函数:

public ArrayList(int initialCapacity)

现以构造一个拥有100万元素的List为例,当使用默认初始化大小时,其消耗的相对时间为125ms左右,当直接制定数组大小为100万时,构造相同的ArrayList仅相对耗时16ms。

遍历列表

遍历列表操作是最常用的列表操作之一,在jdk1.5之后,至少有3中常用的列表遍历方式:

● forEach操作

● 迭代器

● for循环。

String tmp;
long start=System.currentTimeMills();    //ForEach 
for(String s:list){
    tmp=s;
}
System.out.println("foreach spend:"+(System.currentTimeMills()-start));
start = System.currentTimeMills();
for(Iterator it=list.iterator();it.hasNext();){    
   tmp=it.next();
}
System.out.println("Iterator spend;"+(System.currentTimeMills()-start));
start=System.currentTimeMills();
int size=;list.size();
for(int i=0;i

构造一个拥有100万数据的ArrayList和等价的LinkedList,使用以上代码进行测试,测试结果:

1.jpg

什么情况用ArrayList or LinkedList呢?

可以看到,最简便的ForEach循环并没有很好的性能表现,综合性能不如普通的迭代器,而是用for循环通过随机访问遍历列表时,ArrayList表项很好,但是LinkedList的表现却无法让人接受,甚至没有办法等待程序的结束。这是因为对LinkedList进行随机访问时,总会进行一次列表的遍历操作。性能非常差,应避免使用。

总结

ArrayList和LinkedList在性能上各有优缺点,都有各自所适用的地方,总的说来可以描述如下:

1.对ArrayList和LinkedList而言,在列表末尾增加一个元素所花的开销都是固定的。

对ArrayList而言,主要是在内部数组中增加一项,指向所添加的元素,偶尔可能会导致对数组重新进行分配;

而对LinkedList而言,这个开销是统一的,分配一个内部Entry对象。

2.在ArrayList的中间插入或删除一个元素意味着这个列表中剩余的元素都会被移动;而在LinkedList的中间插入或删除一个元素的开销是固定的。

3.LinkedList不支持高效的随机元素访问。

4.ArrayList的空间浪费主要体现在在list列表的结尾预留一定的容量空间,而LinkedList的空间花费则体现在它的每一个元素都需要消耗相当的空间

本文来自编程界网,java教程栏目,欢迎学习

--结束END--

本文标题: java中什么情况下使用ArrayList和LinkedList?

本文链接: https://www.lsjlt.com/news/4188.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

本篇文章演示代码以及资料文档资料下载

下载Word文档到电脑,方便收藏和打印~

下载Word文档
猜你喜欢
  • java中什么情况下使用ArrayList和LinkedList?
    什么情况用ArrayList or LinkedList呢ArrayList 和 LinkedList 是 Java 集合框架中用来存储对象引用列表的两个类。ArrayList 和 LinkedList 都实现 List 接口。先对List...
    99+
    2016-08-06
    java教程 ArrayList LinkedList
  • java中arraylist和linkedlist有什么区别
    ArrayList和LinkedList都是Java中常用的集合类,它们的主要区别如下: 底层数据结构不同:ArrayList底...
    99+
    2023-10-26
    java
  • 在Java中ArrayList和LinkedList的区别是什么
    Java中ArrayList和LinkedList的区别:ArrrayList数据结构是数组,支持随机访问,而 LinkedList数据结构是双向循环链表,不支持随机访问。ArrayList比LinkedList在随机访问的时候效率要高。A...
    99+
    2022-10-15
  • java闭包在什么情况下使用
    闭包在以下情况下使用: 当一个方法需要访问外部作用域的变量时,可以使用闭包。闭包可以将变量保存在函数内部,并且可以在函数外部访问...
    99+
    2023-10-27
    java
  • 什么情况下不能使用Java泛型
    这篇文章主要讲解了“什么情况下不能使用Java泛型”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“什么情况下不能使用Java泛型”吧!1. 前言Java 1.5 引入了泛型来保证类型...
    99+
    2023-06-02
  • 什么情况下需要使用redis
    这篇文章给大家分享的是有关什么情况下需要使用redis的内容。小编觉得挺实用的,因此分享给大家做个参考,一起跟随小编过来看看吧。redis是一个key-value存储系统。和Memcached类似,它支持存...
    99+
    2022-10-18
  • 什么情况下停止使用golang
    在现代编程领域中,Golang (又称 Go) 已经成为一个热门的编程语言。Golang 的设计目标是为了创建高效、可读性强、易于维护和编写的编程语言。它让编程变得更加简单,且在大规模高并发的应用程序开发上表现良好。然而,我们也需要看到问题...
    99+
    2023-05-14
  • 什么情况下使用Adapter模式
    本篇内容主要讲解“什么情况下使用Adapter模式”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“什么情况下使用Adapter模式”吧!适用性以下情况使用Adapter模式• 你想使用一个已经存在...
    99+
    2023-06-19
  • 在什么情况下使用Builder模式
    今天小编给大家分享一下在什么情况下使用Builder模式的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一下吧。适用性在以下情况使用B...
    99+
    2023-06-19
  • web开发中什么情况下使用table或css
    这篇文章主要介绍web开发中什么情况下使用table或css,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完!一、在以下情况下应该尽量使用css布局 1,浏览量大的页面:比如首页。CSS代码少,结构紧凑,加载快,可以大大...
    99+
    2023-06-08
  • 什么情况下需要使用SPFA 算法
    什么情况下需要使用SPFA 算法?相信很多没有经验的人对此束手无策,为此本文总结了问题出现的原因和解决方法,通过这篇文章希望你能解决这个问题。适用范围:给定的图存在负权边,这时类似Dijkstra等算法便没有了用武之地,而Bellman-F...
    99+
    2023-05-31
    spfa算法 情况下
  • java项目中什么情况下HashCode会出现重复
    java项目中什么情况下HashCode会出现重复?针对这个问题,这篇文章详细介绍了相对应的分析和解答,希望可以帮助更多想解决这个问题的小伙伴找到更简单易行的方法。java的缺省算法:  public int&...
    99+
    2023-05-31
    java hashcode 情况下
  • 网站在什么情况下使用VPS主机
    本篇文章给大家分享的是有关网站在什么情况下使用VPS主机,小编觉得挺实用的,因此分享给大家学习,希望大家阅读完这篇文章后可以有所收获,话不多说,跟着小编一起来看看吧。主机行业很花俏,他们提供了大量的选择,以适应用户千奇百怪的需求。但和其他事...
    99+
    2023-06-12
  • 在什么情况下会使用mysql临时表
    这篇文章主要介绍在什么情况下会使用mysql临时表,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完!MySQL在很多情况下都会用到临时表,总结一下什么时候会用到临时表:什么是临时表:My...
    99+
    2022-10-18
  • 在什么情况下可以使用Singleton模式
    这篇文章主要介绍“在什么情况下可以使用Singleton模式”的相关知识,小编通过实际案例向大家展示操作过程,操作方法简单快捷,实用性强,希望这篇“在什么情况下可以使用Singleton模式”文章能帮助大家解决问题。适用性在下面的情况下可以...
    99+
    2023-06-19
  • MySQL数据库中什么情况下需要使用索引
    这篇文章主要介绍MySQL数据库中什么情况下需要使用索引,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完!什么情况下需要使用索引(1)列经常被用于where条件中(2)列中有大...
    99+
    2022-10-18
  • MySQL (InnoDB)在什么情况下无法使用索引
    目录1. 前置知识1.1 InnoDB 索引结构2. 什么情况下不能使用索引2.1 索引列在表达式或函数中【必】2.2 联合索引中,非最左前缀【必】2.3 联合索引中,最左前缀,但是中间有范围查询,那么范围查询后面的列都用不到索引【必】2....
    99+
    2016-04-15
    MySQL (InnoDB)在什么情况下无法使用索引 数据库入门 数据库基础教程
  • Java 中反射的概念、作用和特点,在什么情况下使用反射及使用实例
    反射的概念 反射(Reflection)是 Java 的一种特性,它可以让程序在运行时获取自身的信息,并且动态地操作类或对象的属性、方法和构造器等。通过反射功能,可以让我们在不知道具体类名的情况下,依...
    99+
    2023-10-22
    java
  • MySQL中什么情况下需要用到临时表
    这篇文章主要介绍MySQL中什么情况下需要用到临时表,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完!临时表介绍什么是临时表:MySQL用于存储一些中间结果集的表,临时表只在当前连接可见...
    99+
    2022-10-18
  • VB.NET共享成员需要在什么情况下使用
    本篇内容介绍了“VB.NET共享成员需要在什么情况下使用”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!VB.NET作为一款优秀的面向对象语言...
    99+
    2023-06-17
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作