成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

從面試角度分析LinkedList源碼

開發 前端
LinkedList底層是由雙向鏈表實現的。鏈表好比火車,每節車廂包含了車廂和連接下一節車廂的連接點。而雙向鏈表的每個節點不僅有指向下一個節點的指針,還有指向上一個節點的指針。

[[358239]]

注:本系列文章中用到的jdk版本均為java8

LinkedList類圖如下:

 

LinkedList底層是由雙向鏈表實現的。鏈表好比火車,每節車廂包含了車廂和連接下一節車廂的連接點。而雙向鏈表的每個節點不僅有指向下一個節點的指針,還有指向上一個節點的指針。

在LinkedList源碼中有一個Node靜態類,源碼如下:

  1. private static class Node<E> { 
  2.     E item; 
  3.     Node<E> next
  4.     Node<E> prev; 
  5.  
  6.     Node(Node<E> prev, E element, Node<E> next) { 
  7.         this.item = element; 
  8.         this.next = next
  9.         this.prev = prev; 
  10.     } 

一個Node節點包含三個部分,分別是

  • item:數據
  • next:下一個節點的指針
  • prev:上一個節點的指針

LinkedList的主要變量如下:

  1. // 集合中的元素數量 
  2. transient int size = 0; 
  3.  
  4. /** 
  5.   * 首節點的指針. 
  6.   * Invariant: (first == null && last == null) || 
  7.   *            (first.prev == null && first.item != null
  8.   */ 
  9. transient Node<E> first
  10.  
  11. /** 
  12.   * 尾結點的指針. 
  13.   * Invariant: (first == null && last == null) || 
  14.   *            (last.next == null && last.item != null
  15.   */ 
  16. transient Node<E> last

一、添加元素

LinkedList支持在任意節點位置添加元素,不僅提供了集合常用的add()方法,還提供了addFirst()和addLast(),add()方法默認調用addLast()方法,也就是默認是往鏈表尾部插入元素的。

add()方法源碼:

  1. public boolean add(E e) { 
  2.     linkLast(e); 
  3.     return true

1.1 尾部插入元素

linkLast()源碼如下:

  1. void linkLast(E e) { 
  2.     final Node<E> l = last
  3.     final Node<E> newNode = new Node<>(l, e, null); 
  4.     last = newNode; 
  5.     if (l == null
  6.         first = newNode; 
  7.     else 
  8.         l.next = newNode; 
  9.     size++; 
  10.     modCount++; 

我們來畫張圖演示一下如何給鏈表尾部插入元素:

假如鏈表中沒有元素

對應源碼中的if語句,如果沒有元素則新增的這個節點為鏈表中唯一的一個元素,既是首節點,又是尾結點,前一個元素的指針和后一個元素的指針都是null。這里注意head節點不是第一個節點,head節點只是標識了這個鏈表的地址。

 

假如鏈表中有元素

對應源碼中else語句。先將新增的元素當成Last節點,然后將原來的Last節點的next指向新節點。

  1. else 
  2.     l.next = newNode; 

 

一圖勝前言,畫個圖是不是什么都明白了。

1.2 頭部插入元素

linkFirst()源碼如下:

  1. private void linkFirst(E e) { 
  2.     final Node<E> f = first
  3.     final Node<E> newNode = new Node<>(null, e, f); 
  4.     first = newNode; 
  5.     if (f == null
  6.         last = newNode; 
  7.     else 
  8.         f.prev = newNode; 
  9.     size++; 
  10.     modCount++; 

還是根據上面的圖來解讀一下源碼,先將第一個節點賦值給中間變量f,將新節點newNode賦值給first節點。如果鏈表沒有元素,則Last節點和First節點都是新插入的節點newNode,否則,將原來的First節點的頭指針指向新節點。

二、刪除元素

LinkedList提供的刪除方法有根據索引和元素刪除,除此之外還提供刪除第一個元素和最后一個元素的方法,這里我們只分析一下根據索引刪除的方法。

  1. public E remove(int index) { 
  2.     checkElementIndex(index); 
  3.     return unlink(node(index)); 

checkElementIndex(index)方法就是用來判斷傳輸的索引值是否合法,不合法則拋出數組越界異常。重點來看一下unlink(node(index))方法是如何刪除元素的。

node(index)方法源碼:

node(index)方法就是根據索引獲取該索引位置的節點

  1. Node<E> node(int index) { 
  2.     // assert isElementIndex(index); 
  3.     // 如果指定下標 < 一半元素數量,則從首結點開始遍歷 
  4.     // 否則,從尾結點開始遍歷 
  5.     if (index < (size >> 1)) { 
  6.         Node<E> x = first
  7.         for (int i = 0; i < index; i++) 
  8.             x = x.next
  9.         return x; 
  10.     } else { 
  11.         Node<E> x = last
  12.         for (int i = size - 1; i > index; i--) 
  13.             x = x.prev; 
  14.         return x; 
  15.     } 

unlink(Node x)源碼如下:

  1. E unlink(Node<E> x) { 
  2.     // assert x != null
  3.     final E element = x.item; 
  4.     final Node<E> next = x.next
  5.     final Node<E> prev = x.prev; 
  6.  
  7.     if (prev == null) { 
  8.         first = next
  9.     } else { 
  10.         prev.next = next
  11.         x.prev = null
  12.     } 
  13.  
  14.     if (next == null) { 
  15.         last = prev; 
  16.     } else { 
  17.         next.prev = prev; 
  18.         x.next = null
  19.     } 
  20.  
  21.     x.item = null
  22.     size--; 
  23.     modCount++; 
  24.     return element; 

畫張圖分析一下刪除是如何進行的:

 

  1. 假設刪除的是第一個元素:則它的prev==NULL,我們需要將他的后一個元素(圖中的second)作為第一個元素
  2. 假設刪除的是最后一個元素,則它的next==null,我們需要將他的前一個元素(圖中的second)作為最后一個元素
  3. 如果是中間的任意元素,則需要將它的前一個元素的next指針指向它的后一個元素,同時將它的后一個元素的prev指針指向它的前一個元素。

三、總結

 

LinkedList底層是由雙向鏈表實現的,由于是鏈表實現的,不僅要存放數據,還要存放指針,所以內存開銷要比ArrayList大,刪除元素不需要移動其他元素,只需要改變指針的指向,因此刪除效率更高,同時它沒有實現RandomAccess接口,因此使用迭代器遍歷要比for循環更加高效。LinkedList也支持插入重復值和空值,同樣也是線程不安全的。

本文轉載自微信公眾號「 Java旅途」,可以通過以下二維碼關注。轉載本文請聯系 Java旅途公眾號。

 

責任編輯:武曉燕 來源: Java旅途
相關推薦

2020-12-14 08:03:52

ArrayList面試源碼

2021-07-20 10:26:53

源碼底層ArrayList

2022-03-08 11:29:06

Linux進程系統

2018-10-22 14:28:26

面試官數據公司

2021-03-15 18:47:25

日志開發源碼

2014-05-12 10:06:15

面試測試文化360

2018-04-27 14:46:07

面試簡歷程序員

2023-03-13 07:43:51

PHP類型轉換

2023-09-11 08:51:23

LinkedList雙向鏈表線程

2023-04-14 08:39:01

AQS方法JDK5

2016-11-04 10:30:17

微信小程序

2021-09-10 00:34:22

Java 線程啟動

2021-09-01 17:51:53

技術LinkedList 源碼

2017-04-13 09:45:03

大數據新媒體VC

2017-09-25 16:21:30

Spark on yacluster模式

2021-05-28 11:54:41

區塊鏈數據分析密碼學角度

2020-02-04 09:53:05

數據安全數據泄漏信息安全

2021-12-09 08:31:01

ReentrantLoAQS

2019-04-28 16:10:50

設計Redux前端

2025-03-31 00:00:00

MCPAPI服務器通信
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 国产精品久久久久久中文字 | 操操操操操 | 精品av| 91视视频在线观看入口直接观看 | 色一级| 亚洲国产精品一区二区三区 | 久久新 | 国产精品久久久久久久免费观看 | 欧美在线观看一区 | 日韩一 | 日本小视频网站 | 精品久久网 | 五月天天色 | www.国产一区| 免费啪啪| 天天玩天天操天天干 | 搞av.com| 青青草精品视频 | 人人爽人人爽 | 秋霞电影院午夜伦 | 日本欧美在线观看视频 | 国产精品久久精品 | 国产在线精品一区二区三区 | 欧美久久精品一级c片 | 99爱免费 | 日韩不卡视频在线观看 | 日韩在线不卡视频 | 精品美女视频在线观看免费软件 | 午夜精品久久久久久久久久久久 | 国产一区二区三区视频 | 午夜精品一区二区三区三上悠亚 | 亚洲免费福利视频 | 欧美日韩亚洲国产 | 特黄特色大片免费视频观看 | 日韩无| 91久久国产综合久久91精品网站 | 岛国精品 | 欧美精品一区二区三区在线 | 精品欧美乱码久久久久久1区2区 | 久久久精品综合 | 免费看a|