主頁 > 作業系統 > 線性表的鏈式存盤--單鏈表

線性表的鏈式存盤--單鏈表

2020-09-15 13:43:45 作業系統

Java之線性表的鏈式存盤——單鏈表

我們都知道,線性表的存盤結構分為兩種,順序存盤結構和鏈式存盤結構,線性表的分類可以參考下圖來學習記憶,今天我們主要來學習一下鏈式存盤結構,

一、鏈式存盤介紹

"鏈式存盤結構,地址可以連續也可以不連續的存盤單元存盤資料元素"——來自定義,

其實,你可以想象這樣一個場景,你想找一個人(他的名字叫小譚),于是你首先去問 A , A 說他不知道,但是他說 B 可能知道,并告訴了你 B 在哪里,于是你找到 B ,B 說他不知道,但是他說 C 可能知道,并告訴了你 C 的地址,于是你去找到 C ,C 真的知道小譚在何處,

上面場景其實可以幫助我們去理解鏈表,其實每一個鏈表都包含多個節點,節點又包含兩個部分,一個是資料域(儲存節點含有的資訊),一個是指標域(儲存下一個節點或者上一個節點的地址),而這個指標域就相當于你去問B,B知道C的地址,這個指標域就是存放的 C 的地址,

鏈表下面其實又細分了3種:單鏈表、雙向鏈表和回圈鏈表,今天我們先講單鏈表,

二、單鏈表介紹

什么是單鏈表呢?單鏈表就是每一個節點只有一個指標域的鏈表,如下圖所示,就是一個帶頭節點的單鏈表,下面我們需要知道什么是頭指標,頭節點和首元節點,

頭指標:指向鏈表節點的第一個節點的指標

頭節點:指在鏈表的首元節點之前附設的一個節點

首元節點:指在鏈表中存盤第一個實際資料元素的節點(比如上圖的 a1 節點)

三、單鏈表的創建

單鏈表的創建有兩種方式,分別是頭插法和尾插法,

1、頭插法

頭插法,顧名思義就是把新元素插入到頭部的位置,每次新加的元素都作為鏈表的第一個節點,那么頭插入法在Java中怎么實作呢,首先我們需要定義一個節點,如下

public class ListNode {
  public int val; //資料域
  public ListNode next;//指標域
}

然后我們就創建一個頭指標(不帶頭節點)

//元素個數
int n = 5;
//創建一個頭指標
ListNode headNode = new ListNode();
//頭插入法
headNode= createHead(headNode, n);

然后創建一個私有方法去實作頭插法,這里我們插入5個新元素,頭插入的核心是要先斷開首元節點和頭指標的連接,也就是需要先將原來首元節點的地址存放到新節點的指標域里,也就是 newNode.next = headNode.next,然后再讓頭指標指向新的節點 headNode.next = newNode,這兩步是頭插入的核心,一定要理解,

/**
 * 頭插法
 * 新的節點放在頭節點的后面,之前的就放在新節點的后面
 * @param headNode 頭指標
 * @return
 */
private static ListNode createHead(ListNode headNode, int n) {
  //插入5個新節點
  for (int i = 1; i <= n; i++) {
    ListNode newNode = new ListNode();
    newNode.val = i;
    //將之前的所有節點指向新的節點(也就是新節點指向之前的所有節點)
    newNode.next = headNode.next;
    //將頭指標指向新的節點
    headNode.next = newNode;
  }
  return headNode;
}

最后我把鏈表列印輸出一下(其實也是單鏈表的遍歷),判斷條件就是只有當指標域為空的時候才是最后一個節點,

private static void printLinkedList(ListNode headNode) {
  int countNode = 0;
  while (headNode.next != null){
    countNode++;
    System.out.println(headNode.next.val);
    headNode = headNode.next;
  }
  System.out.println("該單鏈表的節點總數:" +countNode);
}

最后的輸出結果顯然是逆序,因為沒一個新的元素都是從頭部插入的,自然第一個就是最后一個,最后一個就是第一個:

2、尾插法

尾插法,顧名思義就是把新元素插入到尾部的位置(也就是最后一個位置),每次新加的元素都作為鏈表的第最后節點,那么尾插法在 Java 中怎么實作呢,這里還是采用不帶頭節點的實作方式,頭節點和頭指標和頭插入的實作方式一樣,這里我就直接將如何實作:

/**
 * 尾插法
 * 找到鏈表的末尾結點,把新添加的資料作為末尾結點的后續結點
 * @param headNode
 */
private static ListNode createByTail(ListNode headNode, int n) {
  //讓尾指標也指向頭指標
  ListNode tailNode = headNode;
  for (int i = 1; i <= n; i++) {
    ListNode newNode = new ListNode();
    newNode.val = i;
    newNode.next = null;

    //插入到鏈表尾部
    tailNode.next = newNode;
    //指向新的尾節點,tailer永遠存盤最后一個節點的地址
    tailNode = newNode;

  }
  return headNode;
}

和頭插入不同的是,我們需要宣告一個尾指標來輔助我們實作,最開始,尾指標指向頭指標,每插入一個元素,尾指標就后移一下,這里我們來講一下原理:每次往末尾新加一個節點,我們就需要把原來的連接斷開,那怎么斷開呢,我們首先需要讓尾指標指向新的節點,也就是 tailNode.next = newNode; 然后再讓尾指標后移一個位置,讓尾指標指向最后一個節點,也就是尾指標始終指向最后一個節點,最后將頭指標回傳,輸出最后結果:

四、單鏈表的洗掉

既然單鏈表創建好了,怎么在鏈表里面洗掉元素呢,單鏈表的洗掉,我分為了兩種情況洗掉,分別是洗掉第i個節點和洗掉指定元素的節點,

1、洗掉第i個節點

我們可以先來理一下思路:在單鏈表里,節點與節點之間都是通過指標域鏈接起來的,所以如果我們想實作洗掉的操作,實際上是需要我們去改變相應指標域對應得地址的,當想去洗掉第i個元素的時候,比如要洗掉上圖的第3個元素(也就是3),實際上我們要做的就是要讓2號元素指向4號元素(其實就是需要修改2號元素的指標域,讓2號元素的指標域存盤4號元素),那么怎么做才能實作這一步呢?很顯然,要實作這個步驟,我們必須要找到4號元素和2號元素,但是再仔細想一下,其實我們只需要找到2號元素就可以了,因為4號元素的地址存盤再2號的下一個元素的指標域里面,

所以綜上所述分析我們可以得出洗掉的兩個核心步驟:

1.洗掉第i個節點,需要先找到第 i-1 個個節點,也就是第i個節點的前一個節點;

2.然后讓第 i-1 個節點指向第 i-1 個節點的下下個節點

下面的代碼具體實作了怎么洗掉第i個元素,

/**
 * 洗掉第i個節點
 * 1,2 4,4,5
 * 洗掉之后應該是1,2,4,5
 * @param headNode
 * @param index
 * @return
 */
public static ListNode deleteNodeByIndex(ListNode headNode, int index) {
  int count = 1;
  //將參考給它
  ListNode preNode = headNode;
  //看計數器是不是到了i-1,如果到了i-1,就找到了第i-1個節點
  while (preNode.next != null && count <= index -1){
    //尋找要洗掉的當前節點的前一個節點
    count++;
    preNode = preNode.next;
  }
  if (preNode != null){
    preNode.next = preNode.next.next;
  }
  return headNode;
}

2、洗掉指定元素的那個節點

洗掉指定元素節點的實作方法有兩種,第一種就是先找到指定元素對應的鏈表的位置( index ),然后再按照洗掉第 i 個節點的思路洗掉即可,實作方法如下圖所示:

/**
 * 洗掉鏈表指定數值的節點
 * @param headNode
 * @param val
 * @return
 */
private static ListNode deleteNodeByNum(ListNode headNode, int val) {
  ListNode deleteOne = headNode;
  int countByDeleteOne = 1;
  while (deleteOne.next != null){
    if (deleteOne.next.val == val){
      deleteOne = deleteOne.next;
      break;
    }
    countByDeleteOne ++;
    deleteOne = deleteOne.next;
  }
  return deleteNodeByIndex(headNode, countByDeleteOne);
}

第二種方法的實作就很精妙(前提是此節點不是尾節點)

public void deleteNode(ListNode node) {
  //洗掉node即通過將后面的值賦給node,然后更改node的指標指向下下一個結點即可
  node.val = node.next.val;
  node.next = node.next.next;
}

五、單鏈表的查詢(及修改)

單鏈表的查詢實作很簡單,就是遍歷當前單鏈表,然后用一個計數器累加到當前下標,那么當前的這個節點就是要查詢的那個節點,然后再回傳即可,當然需要判斷傳過來的這個下標是否合法,當然如果需要修改,就需要把當前找到的節點的資料域重新賦上需要修改的值即可,這里就不上代碼了,具體實作如下:

private static ListNode searchLinkedList(ListNode headNode, int index) {
  //如果下標是不合法的下標就表示找不到
  if (index < 1 || index > getLinkedListLength(headNode)){
      return null;
  }
  for (int i = 0; i < index; i++) {
    headNode = headNode.next;
  }
  return headNode;
}

獲取單鏈表的長度(注意我這里定義的 headNode 是頭指標不是頭節點)

/**
 * 求單鏈表長度
 * @param headNode
 * @return
 */
private static int getLinkedListLength(ListNode headNode) {
  int countNode = 0;
  while (headNode.next != null){
    countNode++;
    headNode = headNode.next;
  }
 return countNode;
}

六、小結

單鏈表的相關操作就講解完了,其實通過上面對單鏈表的相關操作,我們不難發現,單鏈表的洗掉和插入其實很方便,只需要改變指標的指向就可以完成,但是查找元素的時候就比較麻煩,因為在查找的時候,需要把整個鏈表從頭到尾遍歷一次,


公眾號:良許Linux

有識訓?希望老鐵們來個三連擊,給更多的人看到這篇文章

轉載請註明出處,本文鏈接:https://www.uj5u.com/caozuo/47058.html

標籤:Linux

上一篇:Linux下9種優秀的代碼比對工具推薦

下一篇:Linux下cp的命令解釋

標籤雲
其他(157675) Python(38076) JavaScript(25376) Java(17977) C(15215) 區塊鏈(8255) C#(7972) AI(7469) 爪哇(7425) MySQL(7132) html(6777) 基礎類(6313) sql(6102) 熊猫(6058) PHP(5869) 数组(5741) R(5409) Linux(5327) 反应(5209) 腳本語言(PerlPython)(5129) 非技術區(4971) Android(4554) 数据框(4311) css(4259) 节点.js(4032) C語言(3288) json(3245) 列表(3129) 扑(3119) C++語言(3117) 安卓(2998) 打字稿(2995) VBA(2789) Java相關(2746) 疑難問題(2699) 细绳(2522) 單片機工控(2479) iOS(2429) ASP.NET(2402) MongoDB(2323) 麻木的(2285) 正则表达式(2254) 字典(2211) 循环(2198) 迅速(2185) 擅长(2169) 镖(2155) 功能(1967) .NET技术(1958) Web開發(1951) python-3.x(1918) HtmlCss(1915) 弹簧靴(1913) C++(1909) xml(1889) PostgreSQL(1872) .NETCore(1853) 谷歌表格(1846) Unity3D(1843) for循环(1842)

熱門瀏覽
  • CA和證書

    1、在 CentOS7 中使用 gpg 創建 RSA 非對稱密鑰對 gpg --gen-key #Centos上生成公鑰/密鑰對(存放在家目錄.gnupg/) 2、將 CentOS7 匯出的公鑰,拷貝到 CentOS8 中,在 CentOS8 中使用 CentOS7 的公鑰加密一個檔案 gpg -a ......

    uj5u.com 2020-09-10 00:09:53 more
  • Kubernetes K8S之資源控制器Job和CronJob詳解

    Kubernetes的資源控制器Job和CronJob詳解與示例 ......

    uj5u.com 2020-09-10 00:10:45 more
  • VMware下安裝CentOS

    VMware下安裝CentOS 一、軟硬體準備 1 Centos鏡像準備 1.1 CentOS鏡像下載地址 下載地址 1.2 CentOS鏡像下載程序 點擊下載地址進入如下圖的網站,選擇需要下載的版本,這里選擇的是Centos8,點擊如圖所示。 決定選擇Centos8后,選擇想要的鏡像源進行下載,此 ......

    uj5u.com 2020-09-10 00:12:10 more
  • 如何使用Grep命令查找多個字串

    如何使用Grep 命令查找多個字串 大家好,我是良許! 今天向大家介紹一個非常有用的技巧,那就是使用 grep 命令查找多個字串。 簡單介紹一下,grep 命令可以理解為是一個功能強大的命令列工具,可以用它在一個或多個輸入檔案中搜索與正則運算式相匹配的文本,然后再將每個匹配的文本用標準輸出的格式 ......

    uj5u.com 2020-09-10 00:12:28 more
  • git配置http代理

    git配置http代理 經常遇到克隆 github 慢的問題,這里記錄一下幾種配置 git 代理的方法,解決 clone github 過慢。 目錄 git配置代理 git單獨配置github代理 git配置全域代理 配置終端環境變數 git配置代理 主要使用 git config 命令 git單獨 ......

    uj5u.com 2020-09-10 00:12:33 more
  • Linux npm install 裝包時提示Error EACCES permission denied解

    npm install 裝包時提示Error EACCES permission denied解決辦法 ......

    uj5u.com 2020-09-10 00:12:53 more
  • Centos 7下安裝nginx,使用yum install nginx,提示沒有可用的軟體包

    Centos 7下安裝nginx,使用yum install nginx,提示沒有可用的軟體包。 18 (flaskApi) [root@67 flaskDemo]# yum -y install nginx 19 已加載插件:fastestmirror, langpacks 20 Loading ......

    uj5u.com 2020-09-10 00:13:13 more
  • Linux查看服務器暴力破解ssh IP

    在公網的服務器上經常遇到別人爆破你服務器的22埠,用來挖礦或者干其他嘿嘿嘿的事情~ 這種情況下正確的做法是: 修改默認ssh的22埠 使用設定密鑰登錄或者白名單ip登錄 建議服務器密碼為復雜密碼 創建普通用戶登錄服務器(root權限過大) 建立堡壘機,實作統一管理服務器 統計爆破IP [root ......

    uj5u.com 2020-09-10 00:13:17 more
  • CentOS 7系統常見快捷鍵操作方式

    Linux系統中一些常見的快捷方式,可有效提高操作效率,在某些時刻也能避免操作失誤帶來的問題。 ......

    uj5u.com 2020-09-10 00:13:31 more
  • CentOS 7作業系統目錄結構介紹

    作業系統存在著大量的資料檔案資訊,相應檔案資訊會存在于系統相應目錄中,為了更好的管理資料資訊,會將系統進行一些目錄規劃,不同目錄存放不同的資源。 ......

    uj5u.com 2020-09-10 00:13:35 more
最新发布
  • vim的常用命令

    Vim的6種基本模式 1. 普通模式在普通模式中,用的編輯器命令,比如移動游標,洗掉文本等等。這也是Vim啟動后的默認模式。這正好和許多新用戶期待的操作方式相反(大多數編輯器默認模式為插入模式)。 2. 插入模式在這個模式中,大多數按鍵都會向文本緩沖中插入文本。大多數新用戶希望文本編輯器編輯程序中一 ......

    uj5u.com 2023-04-20 08:43:21 more
  • vim的常用命令

    Vim的6種基本模式 1. 普通模式在普通模式中,用的編輯器命令,比如移動游標,洗掉文本等等。這也是Vim啟動后的默認模式。這正好和許多新用戶期待的操作方式相反(大多數編輯器默認模式為插入模式)。 2. 插入模式在這個模式中,大多數按鍵都會向文本緩沖中插入文本。大多數新用戶希望文本編輯器編輯程序中一 ......

    uj5u.com 2023-04-20 08:42:36 more
  • docker學習

    ###Docker概述 真實專案部署環境可能非常復雜,傳統發布專案一個只需要一個jar包,運行環境需要單獨部署。而通過Docker可將jar包和相關環境(如jdk,redis,Hadoop...)等打包到docker鏡像里,將鏡像發布到Docker倉庫,部署時下載發布的鏡像,直接運行發布的鏡像即可。 ......

    uj5u.com 2023-04-19 09:26:53 more
  • 設定Windows主機的瀏覽器為wls2的默認瀏覽器

    這里以Chrome為例。 1. 準備作業 wsl是可以使用Windows主機上安裝的exe程式,出于安全考慮,默認情況下改功能是無法使用。要使用的話,終端需要以管理員權限啟動。 我這里以Windows Terminal為例,介紹如何默認使用管理員權限打開終端,具體操作如下圖所示: 2. 操作 wsl ......

    uj5u.com 2023-04-19 09:25:49 more
  • docker學習

    ###Docker概述 真實專案部署環境可能非常復雜,傳統發布專案一個只需要一個jar包,運行環境需要單獨部署。而通過Docker可將jar包和相關環境(如jdk,redis,Hadoop...)等打包到docker鏡像里,將鏡像發布到Docker倉庫,部署時下載發布的鏡像,直接運行發布的鏡像即可。 ......

    uj5u.com 2023-04-19 09:19:04 more
  • Linux學習筆記

    IP地址和主機名 IP地址 ifconfig可以用來查詢本機的IP地址,如果不能使用,可以通過install net-tools安裝。 Centos系統下ens33表示主網卡;inet后表示IP地址;lo表示本地回環網卡; 127.0.0.1表示代指本機;0.0.0.0可以用于代指本機,同時在放行設 ......

    uj5u.com 2023-04-18 06:52:01 more
  • 解決linux系統的kdump服務無法啟動的問題

    問題:專案麒麟系統服務器的kdump服務無法啟動,沒有相關日志無法定位問題。 1、查看服務狀態是關閉的,重啟系統也無法啟動 systemctl status kdump 2、修改grub引數,修改“crashkernel”為“512M(有的機器數值太大太小都會導致報錯,建議從128M開始試,或者加個 ......

    uj5u.com 2023-04-12 09:59:50 more
  • 解決linux系統的kdump服務無法啟動的問題

    問題:專案麒麟系統服務器的kdump服務無法啟動,沒有相關日志無法定位問題。 1、查看服務狀態是關閉的,重啟系統也無法啟動 systemctl status kdump 2、修改grub引數,修改“crashkernel”為“512M(有的機器數值太大太小都會導致報錯,建議從128M開始試,或者加個 ......

    uj5u.com 2023-04-12 09:59:01 more
  • 你是不是暴露了?

    作者:袁首京 原創文章,轉載時請保留此宣告,并給出原文連接。 如果您是計算機相關從業人員,那么應該經歷不止一次網路安全專項檢查了,你肯定是收到過資訊系統技術檢測報告,要求你加強風險監測,確保你提供的系統服務堅實可靠了。 沒檢測到問題還好,檢測到問題的話,有些處理起來還是挺麻煩的,尤其是線上正在運行的 ......

    uj5u.com 2023-04-05 16:52:56 more
  • 細節拉滿,80 張圖帶你一步一步推演 slab 記憶體池的設計與實作

    1. 前文回顧 在之前的幾篇記憶體管理系列文章中,筆者帶大家從宏觀角度完整地梳理了一遍 Linux 記憶體分配的整個鏈路,本文的主題依然是記憶體分配,這一次我們會從微觀的角度來探秘一下 Linux 內核中用于零散小記憶體塊分配的記憶體池 —— slab 分配器。 在本小節中,筆者還是按照以往的風格先帶大家簡單 ......

    uj5u.com 2023-04-05 16:44:11 more