日韩无码专区无码一级三级片|91人人爱网站中日韩无码电影|厨房大战丰满熟妇|AV高清无码在线免费观看|另类AV日韩少妇熟女|中文日本大黄一级黄色片|色情在线视频免费|亚洲成人特黄a片|黄片wwwav色图欧美|欧亚乱色一区二区三区

RELATEED CONSULTING
相關(guān)咨詢
選擇下列產(chǎn)品馬上在線溝通
服務(wù)時(shí)間:8:30-17:00
你可能遇到了下面的問題
關(guān)閉右側(cè)工具欄

新聞中心

這里有您想知道的互聯(lián)網(wǎng)營銷解決方案
從面試角度分析LinkedList源碼

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

LinkedList類圖如下:

LinkedList底層是由雙向鏈表實(shí)現(xiàn)的。鏈表好比火車,每節(jié)車廂包含了車廂和連接下一節(jié)車廂的連接點(diǎn)。而雙向鏈表的每個(gè)節(jié)點(diǎn)不僅有指向下一個(gè)節(jié)點(diǎn)的指針,還有指向上一個(gè)節(jié)點(diǎn)的指針。

在LinkedList源碼中有一個(gè)Node靜態(tài)類,源碼如下:

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

一個(gè)Node節(jié)點(diǎn)包含三個(gè)部分,分別是

  • item:數(shù)據(jù)
  • next:下一個(gè)節(jié)點(diǎn)的指針
  • prev:上一個(gè)節(jié)點(diǎn)的指針

LinkedList的主要變量如下:

 
 
 
 
  1. // 集合中的元素?cái)?shù)量 
  2. transient int size = 0; 
  3.  
  4. /** 
  5.   * 首節(jié)點(diǎn)的指針. 
  6.   * Invariant: (first == null && last == null) || 
  7.   *            (first.prev == null && first.item != null) 
  8.   */ 
  9. transient Node first; 
  10.  
  11. /** 
  12.   * 尾結(jié)點(diǎn)的指針. 
  13.   * Invariant: (first == null && last == null) || 
  14.   *            (last.next == null && last.item != null) 
  15.   */ 
  16. transient Node last; 

一、添加元素

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

add()方法源碼:

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

1.1 尾部插入元素

linkLast()源碼如下:

 
 
 
 
  1. void linkLast(E e) { 
  2.     final Node l = last; 
  3.     final Node 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++; 

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

假如鏈表中沒有元素

對應(yīng)源碼中的if語句,如果沒有元素則新增的這個(gè)節(jié)點(diǎn)為鏈表中唯一的一個(gè)元素,既是首節(jié)點(diǎn),又是尾結(jié)點(diǎn),前一個(gè)元素的指針和后一個(gè)元素的指針都是null。這里注意head節(jié)點(diǎn)不是第一個(gè)節(jié)點(diǎn),head節(jié)點(diǎn)只是標(biāo)識了這個(gè)鏈表的地址。

假如鏈表中有元素

對應(yīng)源碼中else語句。先將新增的元素當(dāng)成Last節(jié)點(diǎn),然后將原來的Last節(jié)點(diǎn)的next指向新節(jié)點(diǎn)。

 
 
 
 
  1. else 
  2.     l.next = newNode; 

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

1.2 頭部插入元素

linkFirst()源碼如下:

 
 
 
 
  1. private void linkFirst(E e) { 
  2.     final Node f = first; 
  3.     final Node 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++; 

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

二、刪除元素

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

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

checkElementIndex(index)方法就是用來判斷傳輸?shù)乃饕凳欠窈戏ǎ缓戏▌t拋出數(shù)組越界異常。重點(diǎn)來看一下unlink(node(index))方法是如何刪除元素的。

node(index)方法源碼:

node(index)方法就是根據(jù)索引獲取該索引位置的節(jié)點(diǎn)

 
 
 
 
  1. Node node(int index) { 
  2.     // assert isElementIndex(index); 
  3.     // 如果指定下標(biāo) < 一半元素?cái)?shù)量,則從首結(jié)點(diǎn)開始遍歷 
  4.     // 否則,從尾結(jié)點(diǎn)開始遍歷 
  5.     if (index < (size >> 1)) { 
  6.         Node x = first; 
  7.         for (int i = 0; i < index; i++) 
  8.             x = x.next; 
  9.         return x; 
  10.     } else { 
  11.         Node 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 x) { 
  2.     // assert x != null; 
  3.     final E element = x.item; 
  4.     final Node next = x.next; 
  5.     final Node 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; 

畫張圖分析一下刪除是如何進(jìn)行的:

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

三、總結(jié)

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

本文轉(zhuǎn)載自微信公眾號「 Java旅途」,可以通過以下二維碼關(guān)注。轉(zhuǎn)載本文請聯(lián)系 Java旅途公眾號。


分享名稱:從面試角度分析LinkedList源碼
鏈接地址:http://www.5511xx.com/article/cccgcge.html