jdk源碼->集合->LinkedList

来源:https://www.cnblogs.com/kundeg/archive/2018/01/30/8386953.html
-Advertisement-
Play Games

類的屬性 構造函數 LinkedList()型構造函數 LinkedList(Collection)型構造函數 核心函數分析 add(E e)函數 linklast(E e)函數分析 addLast函數的實際調用 linkFirst(E e)函數 addFirst函數的實際調用 add(int in ...


類的屬性

public class LinkedList<E>
    extends AbstractSequentialList<E>
    implements List<E>, Deque<E>, Cloneable, java.io.Serializable
{
    // 實際元素個數
    transient int size = 0;
    // 指向頭節點的指針
    transient Node<E> first;
    // 指向尾節點的指針
    transient Node<E> last;
}    

構造函數

LinkedList()型構造函數 

public LinkedList() {
}

LinkedList(Collection<? extends E>)型構造函數  

public LinkedList(Collection<? extends E> c) {
        // 調用無參構造函數
        this();
        // 添加集合中所有的元素
        addAll(c);
    }

核心函數分析

add(E e)函數

public boolean add(E e) {
        // 添加到末尾
        linkLast(e);
        return true;
    }

linklast(E e)函數分析 addLast函數的實際調用

    void linkLast(E e) {
        // 記住尾結點指針
        final Node<E> l = last;
        // 新結點的prev為l,後繼為null,值為e
        final Node<E> newNode = new Node<>(l, e, null);
        // 更新尾結點指針
        last = newNode;    
        if (l == null) // 如果鏈表為空
            first = newNode; // 頭指針等於尾指針
        else // 尾結點不為空
            l.next = newNode; // 尾結點的後繼為新生成的結點
        // 大小加1    
        size++;
        // 結構性修改加1
        modCount++;
    }

linkFirst(E e)函數 addFirst函數的實際調用

    private void linkFirst(E e) {
        //記住頭結點
        final Node<E> f = first;
        //實例化新節點,prev = null,last = first
        final Node<E> newNode = new Node<>(null, e, f);
        //頭指針指向新節點
        first = newNode;
        if (f == null)
            //如果頭指針為空的話,尾指針也指向新節點
            last = newNode;
        else
            f.prev = newNode;
        size++;
        modCount++;
    }

add(int index,E element)函數

    public void add(int index, E element) {
        checkPositionIndex(index);

        if (index == size)
            linkLast(element);
        else
            linkBefore(element, node(index));
    }

linkBefore(E e,Node succ)

   void linkBefore(E e, Node<E> succ) {
        // 記住前驅
        final Node<E> pred = succ.prev;
        //實例化新節點,初始化前驅,後繼
        final Node<E> newNode = new Node<>(pred, e, succ);
        succ.prev = newNode;
        if (pred == null)
            //如果為頭插,修改頭指針指向該新節點
            first = newNode;
        else
            pred.next = newNode;
        size++;
        modCount++;
    }

addAll(int index, Collection<? extends E> c)函數

    // 添加一個集合
    public boolean addAll(int index, Collection<? extends E> c) {
        // 檢查插入的的位置是否合法
        checkPositionIndex(index);
        // 將集合轉化為數組
        Object[] a = c.toArray();
        // 保存集合大小
        int numNew = a.length;
        if (numNew == 0) // 集合為空,直接返回
            return false;

        Node<E> pred, succ; // 前驅,後繼,註意是插入整個集合的入前驅與後繼
        if (index == size) { // 如果插入位置為鏈表末尾,則後繼為null,前驅為尾結點
            succ = null;
            pred = last;
        } else { // 插入位置為其他某個位置
            succ = ode(index); n// 尋找到該結點
            pred = succ.prev; // 保存該結點的前驅
        }

        for (Object o : a) { // 遍曆數組
            E e = (E) o; // 向下轉型
            // 生成新結點
            Node<E> newNode = new Node<>(pred, e, null);
            if (pred == null) // 表示在第一個元素之前插入(索引為0的結點)
                first = newNode;
            else
                pred.next = newNode;
            pred = newNode;
        }

        if (succ == null) { // 表示在最後一個元素之後插入
            last = pred;
        } else {
            pred.next = succ;
            succ.prev = pred;
        }
        // 修改實際元素個數
        size += numNew;
        // 結構性修改加1
        modCount++;
        return true;
    }

node(int index)函數

Node<E> node(int index) {
        // 判斷插入的位置在鏈表前半段或者是後半段
        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; // 返回該結點
        }
    }

為什麼不分成三段,因為鏈表的取index結點依賴於first和last指針,即使分成再多的段,定位多麼的精確,還是要從first或者last開始遍歷

remove(Object o)函數

    public boolean remove(Object o) {
        if (o == null) {
            //同樣說明LinkedList允許插入null
            for (Node<E> x = first; x != null; x = x.next) {
                if (x.item == null) {
                    unlink(x);
                    return true;
                }
            }
        } else {
            for (Node<E> x = first; x != null; x = x.next) {
                if (o.equals(x.item)) {
                    unlink(x);
                    return true;
                }
            }
        }
        return false;
    }

unlink(Node x)函數

    E unlink(Node<E> x) {
        //返回x的結點值
        final E element = x.item;
        //記住前驅
        final Node<E> next = x.next;
        //記住後繼
        final Node<E> prev = x.prev;
        //如果刪除的結點是頭結點,頭指針指向next
        if (prev == null) {
            first = next;
        } else {
            prev.next = next;
            x.prev = null;
        }
        //如果刪除的結點是尾結點,尾指針指向prev
        if (next == null) {
            last = prev;
        } else {
            next.prev = prev;
            x.next = null;
        }
        //集合內容清null
        x.item = null;
        size--;
        modCount++;
        return element;
    }

unlinkFirst(Node f)函數 removeFirst函數的實際調用

    private E unlinkFirst(Node<E> f) {
        // 記住結點值和後繼
        final E element = f.item;
        final Node<E> next = f.next;
        f.item = null;
        f.next = null; // help GC
        //頭指針指向後繼
        first = next;
        if (next == null)
            //如果刪除結點後鏈表為空,尾指針指向空
            last = null;
        else
            next.prev = null;
        size--;
        modCount++;
        return element;
    }

unlinkLast(Node l) removeLast函數的實際調用

    private E unlinkLast(Node<E> l) {
        // 記住結點值和前驅
        final E element = l.item;
        final Node<E> prev = l.prev;
        l.item = null;
        l.prev = null; // help GC
        //尾結點指向前驅
        last = prev;
        if (prev == null)
            //如果刪除結點後鏈表為空,頭指針指向空
            first = null;
        else
            prev.next = null;
        size--;
        modCount++;
        return element;
    }

您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 使用 getch() 函數,需要先引入 conio.h 頭文件 然而,我使用的是 cygwin 作為編譯環境,找不到 conio.h ,所以只能想辦法找替代方法,或者自己構造一個具有類似功能的函數。 可惜,剛學編程沒多久,一時之間也是沒有想到什麼合適的替代方法,若說自己構造這個函數,這就更難了。 於 ...
  • 1.A 演算法 我們普通的搜索演算法往往複雜度都是指數級,OI中這樣的複雜度無法滿足我們的要求。這時我們一般都會進行一些剪枝優化,但在有些題目中卻可以有更加巧妙的方法——A 演算法。 A 演算法作為一種基礎的啟髮式搜索,它不同於DFS和BFS將所有情況進行遍歷,它能從所有情況中選出較優的再進行遍歷。因此,它 ...
  • 一.使用environ指針輸出環境變數 代碼如下: ~~~~ include include define MAX_INPUT 20 / 引用指針 / extern char environ; int main(int argc, char argv) { char temp = NULL; cha ...
  • 前年學習opengl做的一個小東西。 原本計劃將gpuimage 的演算法一個一個轉寫成cpu版本 c,c++ 版本。 gpuimage 項目參考: https://github.com/BradLarson/GPUImage https://github.com/BradLarson/GPUImag ...
  • @RequestMapping 映射請求1.SpringMVC使用@RequestMapping註解為控制器指定可以處理哪些URL請求。2.在控制器的類定義及方法定義處都可標註@RequestMapping。 (1).類定義處:提供初步的請求映射信息。相對於WEB應用的根目錄。 (2).提供進一步的 ...
  • hello world必備-》print函數 print(): 作用: 列印函數,列印數據到屏幕中 參數列表: print(value, ..., sep=' ', end='\n', file=sys.stdout, flush=False) 參數解釋: value,...,代表可以有多個要列印的... ...
  • 思路:能夠向四個方向前進,同時前進步數不超過m。那麼,在遍歷的時候運用一層迴圈將不超過m步的都一一列舉出來。    同時,在遍歷的過程中需要記錄數據形成記憶是搜索。 ...
  • 近來想用pygame做做游戲,在 xishui 大神的目光博客中學了學這東西,就上一段自己寫的飛機大戰的代碼,主要是對鍵盤控制飛機的移動做了相關的優化 在這裡,飛機的偏移量之所以設置四個而不是兩個,是因為如果設置的是兩個,即控制x和y軸,那麼飛機控制的方向只能是x軸或y軸, 就比如說,當按住a鍵的時 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...