代碼隨想錄算法訓練營Day4|24. 兩兩交換鏈表中的節點、19.刪除鏈表的倒數第N個節點、 142.環形鏈表II、面試題 02.07. 鏈表相交

24. 兩兩交換鏈表中的節點

這道題的關鍵在于:
1、在置換兩個節點的時候,當前節點需要在這倆節點之前一個節點。并且要提前保存cur.next以及cur.next.next。
2、每次置換完一組節點,cur = cur.next.next
3、判斷結束的標志:奇數個節點:cur.next.next != null 偶數個節點:cur.next != null 為了保證cur.next.next不會產生空指針,需要先判斷next != null
4、最終返回的是虛擬節點的下一個節點,因為最初的head節點已經不在鏈表最前面了
具體步驟如下:
在這里插入圖片描述

dummyNode.next = head;//對于0、1節點,需要創建一個虛擬節點.
ListNode temp1 = cur.next;
ListNode temp2 = cur.next.next.next;//并保存temp1和temp2.

①②③步驟按序進行
①cur.next = cur.next.next;
②cur.next.next = temp1;
③ temp1.next = temp2;
在這里插入圖片描述
cur = cur.next.next;

/*** Definition for singly-linked list.* public class ListNode {*     int val;*     ListNode next;*     ListNode() {}*     ListNode(int val) { this.val = val; }*     ListNode(int val, ListNode next) { this.val = val; this.next = next; }* }*/
class Solution {public ListNode swapPairs(ListNode head) {ListNode dummyNode = new ListNode();dummyNode.next = head;ListNode cur = dummyNode;while(cur.next != null && cur.next.next != null){ListNode temp1 = cur.next;ListNode temp2 = cur.next.next.next;cur.next = cur.next.next;cur.next.next = temp1;temp1.next = temp2;cur = cur.next.next;}return dummyNode.next;}
}

19.刪除鏈表的倒數第N個節點

法一:遍歷鏈表得出鏈表長度length,用length - n定位要刪除的節點,缺點是需要第二次遍歷鏈表
代碼:

/*** Definition for singly-linked list.* public class ListNode {*     int val;*     ListNode next;*     ListNode() {}*     ListNode(int val) { this.val = val; }*     ListNode(int val, ListNode next) { this.val = val; this.next = next; }* }*/
class Solution {public ListNode removeNthFromEnd(ListNode head, int n) {int length = 0;ListNode dummyNode = new ListNode();dummyNode.next = head;ListNode cur = dummyNode;while(cur.next != null){length++;cur = cur.next;}ListNode cur2 = dummyNode;for(int i = 0;i < length - n;i++){cur2 = cur2.next;}cur2.next = cur2.next.next;return dummyNode.next;}
}

法二:讓快指針先走n步,然后快慢指針同時走。
這個思想很巧妙也很好理解,關鍵是臨界值的設定很容易出錯,可以用只有一個節點的鏈表刪除倒數第一個節點的特殊情況去寫臨界值。

/*** Definition for singly-linked list.* public class ListNode {*     int val;*     ListNode next;*     ListNode() {}*     ListNode(int val) { this.val = val; }*     ListNode(int val, ListNode next) { this.val = val; this.next = next; }* }*/
class Solution {public ListNode removeNthFromEnd(ListNode head, int n) {ListNode dummyNode = new ListNode();dummyNode.next = head;ListNode fast = dummyNode;ListNode slow = dummyNode;for(int i = 0;i < n-1;i++){fast = fast.next;}//假設只有一個節點的鏈表要刪除倒數第一個節點,n=1,此時快慢節點不需要中間有差值,slow和fast都指向dummyNodewhile(fast.next.next != null){fast = fast.next;slow = slow.next;}//因為fast.next.next == null,fast和slow都不需要往下移動slow.next = slow.next.next;//刪除第一個節點return dummyNode.next;}
}

或者這樣也可以:
這倆區別在于如果在
for(int i = 0;i < n;i++){
fast = fast.next;
}中i移動的條件為這個,fast為向下移動一位。那么響應的下面
while(fast.next != null){
fast = fast.next;
slow = slow.next;
}中while的判定條件由fast.next.next!= null變為fast.next != null

/*** Definition for singly-linked list.* public class ListNode {*     int val;*     ListNode next;*     ListNode() {}*     ListNode(int val) { this.val = val; }*     ListNode(int val, ListNode next) { this.val = val; this.next = next; }* }*/
class Solution {public ListNode removeNthFromEnd(ListNode head, int n) {ListNode dummyNode = new ListNode();dummyNode.next = head;ListNode fast = dummyNode;ListNode slow = dummyNode;for(int i = 0;i < n;i++){fast = fast.next;}while(fast.next != null){fast = fast.next;slow = slow.next;}slow.next = slow.next.next;return dummyNode.next;}
}

142.環形鏈表II

這道題老是犯一個錯誤,就是控制fast和slow相遇的循環條件寫成fast != slow,slow和fast就移動,但是要注意一點,slow和fast在初始的時候就應該分別向下移動一位和兩位,以防止while進入死循環。
在這里插入圖片描述
定義slow和fast兩個快慢指針,slow每次移動一位,fast每次移動兩位,直到相遇。slow和fast走過的路程分別為
在這里插入圖片描述
fast是slow的兩倍,可以得到等式,最終得到x和z、y的關系式
在這里插入圖片描述
這個時候slow和fast在同一位置,再定義一個res節點指針指向head,res和slow指針同時一位一位向下移動,當兩個指針相遇的時候,這個節點就是環形的入口。
代碼如下:

/*** Definition for singly-linked list.* class ListNode {*     int val;*     ListNode next;*     ListNode(int x) {*         val = x;*         next = null;*     }* }*/
public class Solution {public ListNode detectCycle(ListNode head) {if(head == null) return null;if(head.next == null || head.next.next == null) return null;ListNode fast = head.next.next;ListNode slow = head.next;while(fast != slow){if(fast.next != null && fast.next.next != null){fast = fast.next.next;}else{return null;}slow = slow.next;}ListNode res = head;while(slow != res){slow = slow.next;res = res.next;}return res;}
}

這里尋找fast和slow相遇節點的while循環可以優化一下:

/*** Definition for singly-linked list.* class ListNode {*     int val;*     ListNode next;*     ListNode(int x) {*         val = x;*         next = null;*     }* }*/
public class Solution {public ListNode detectCycle(ListNode head) {if(head == null) return null;if(head.next == null || head.next.next == null) return null;ListNode fast = head;ListNode slow = head;while(true){if(fast.next != null && fast.next.next != null) fast = fast.next.next;else return null;slow = slow.next;if(slow == fast) break;}ListNode res = head;while(slow != res){slow = slow.next;res = res.next;}return res;}
}

面試題 02.07. 鏈表相交

思路:
先計算出兩條鏈表的長度,再將短的那一條鏈表的指針先移動鏈表長度差值個,然后再一起向下遍歷直到相遇,如果沒有相遇,最終都會到達null,返回停下之后的那個節點即可。

/*** Definition for singly-linked list.* public class ListNode {*     int val;*     ListNode next;*     ListNode(int x) {*         val = x;*         next = null;*     }* }*/
public class Solution {public ListNode getIntersectionNode(ListNode headA, ListNode headB) {ListNode curA = headA;ListNode curB = headB;int lengthA = 0;int lengthB = 0;while(curA != null){curA = curA.next;lengthA++;}while(curB != null){curB = curB.next;lengthB++;}if(lengthA > lengthB){for(int i = 0;i < lengthA-lengthB;i++){headA = headA.next;} }else{for(int i = 0;i < lengthB-lengthA;i++){headB = headB.next;} }while(headA != null && headA != headB){headA = headA.next;headB = headB.next;}return headA;}
}

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:
http://www.pswp.cn/bicheng/15916.shtml
繁體地址,請注明出處:http://hk.pswp.cn/bicheng/15916.shtml
英文地址,請注明出處:http://en.pswp.cn/bicheng/15916.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

如何禁止U盤拷貝文件|禁止U盤使用的軟件有哪些

禁止U盤拷貝文件的方法有很多&#xff0c;比如使用注冊表、組策略編輯器等&#xff0c;但這些方法都適合個人&#xff0c;不適合企業&#xff0c;因為企業需要對下屬多臺電腦進行遠程管控&#xff0c;需要方便、省時、省力的方法。目前來說&#xff0c;最好的方法就是使用第三方…

Unity websocket客戶端

&#x1f3c6; 個人愚見&#xff0c;沒事寫寫筆記 &#x1f3c6;《博客內容》&#xff1a;Unity3D開發內容 &#x1f3c6;&#x1f389;歡迎 &#x1f44d;點贊?評論?收藏 &#x1f50e;目標&#xff1a;服務器和客戶端可以實時的傳輸信息 ??實現目標&#xff1a; 使用的w…

技術速遞|無障礙應用程序之旅:鍵盤可訪問性和 .NET MAUI

作者&#xff1a;Rachel Kang 排版&#xff1a;Alan Wang 首先讓我們一起來看看您的應用程序是否支持鍵盤訪問&#xff1a; 啟動您的其中一個應用。如果您的設備尚未連接物理鍵盤&#xff0c;請連接物理鍵盤。像平常一樣導航您的應用程序&#xff0c;并且僅使用鍵盤來執行此操…

如何使用Rust構建Python原生庫?注意,不是動態鏈接庫!!!

參考文檔&#xff1a;https://github.com/PyO3/pyo3 創建python虛擬環境&#xff1a; conda create --name pyo3 python3.11.7激活虛擬環境&#xff1a; conda activate pyo3安裝依賴&#xff1a; pip install maturin初始化項目&#xff1a; maturin init構建項目&#x…

設計模式--目錄

設計模式是軟件工程中為解決常見問題而總結出來的一系列通用解決方案。它們可以分為三大類別&#xff1a;創建型模式、結構型模式和行為型模式。下面列舉了一些常見的設計模式及其分類&#xff1a; 創建型模式(Creational Patterns) 創建型模式關注對象的創建過程&#xff0c…

小程序checkbox改成圓形與radio樣式保持一致

修改前 修改后 html: <view class"agreement"><checkbox value"{{ isAgreed }}" bind:tap"toggleCheckbox" /><text>我同意室外智能健身房 <text class"link" bind:tap"showUserProtocol">用戶協…

【JTS Topology Suite】Java對二維幾何進行平移、縮放、旋轉等坐標變換

JTS介紹 Github項目地址&#xff1a;https://github.com/locationtech/jts Maven庫地址&#xff1a;https://mvnrepository.com/artifact/org.locationtech.jts JTS Topology Suite是一個用于創建和操作二維矢量幾何的Java庫。 JTS有對應的.NET版本NetTopologySuite庫&…

P3128 [USACO15DEC] Max Flow P題解(樹上差分,最近公共祖先,圖論)

前言&#xff1a; 題目鏈接&#xff1a;P3128 [USACO15DEC] Max Flow P - 洛谷 | 計算機科學教育新生態 (luogu.com.cn) 講解&#xff1a; 這一題含金量真算高的&#xff0c;包含了建樹&#xff08;用了圖論的知識&#xff09;&#xff0c;求最近公共祖先&#xff08;倍增法…

2024目前網上最火短劇機器人做法,自動搜索發劇 自動更新資源 自動分享資源

目前整個項目圈子很多的短劇機器人&#xff0c;我寫的&#xff0c;自動搜索發劇&#xff0c;自動更新資源&#xff0c;自動分享資源&#xff0c;前段時間大部分做短劇的都是做的短劇分成&#xff0c;我的一個學員做的30W播放量才200塊收益&#xff0c;備受啟發&#xff0c;我就…

springboot社區助老志愿服務系統-計算機畢業設計源碼96682

摘要 大數據時代下&#xff0c;數據呈爆炸式地增長。為了迎合信息化時代的潮流和信息化安全的要求&#xff0c;利用互聯網服務于其他行業&#xff0c;促進生產&#xff0c;已經是成為一種勢不可擋的趨勢。在圖書館管理的要求下&#xff0c;開發一款整體式結構的社區助老志愿服務…

社交媒體數據恢復:綠洲

本教程將向您展示如何在綠洲平臺上備份和恢復數據&#xff0c;但不涉及推薦任何具體的數據恢復軟件。 一、綠洲平臺數據備份 為了確保數據的安全&#xff0c;在日常使用過程中&#xff0c;我們需要定期備份綠洲平臺上的數據。以下是備份綠洲平臺數據的步驟&#xff1a; 登錄綠…

three.js能實現啥效果?看過來,這里都是它的菜(09)

Hi&#xff0c;這是第九期了&#xff0c;繼續分享three.js在可視化大屏中的應用&#xff0c;本期分享位移動畫的實現。 位移動畫 Three.js位移動畫是指在Three.js中實現物體位置的平移動畫。通過改變物體的位置屬性&#xff0c;可以實現物體沿著指定路徑從一個位置移動到另一…

Java——圖書管理系統萬字詳解(附代碼)

框架搭建 book包 將書相關的放到book包中&#xff0c;創建一個Book類用來設置書的屬性&#xff0c;包括書名、作者、價格、類型、是否被借出等。 以上屬性均被private所修飾 利用編譯器生成構造方法&#xff08;不需要構造isBorrowed&#xff0c;因為其初始值為false&#…

微前端架構 之 應用之間樣式隔離 (四)

1. 使用 CSS Modules 進行樣式隔離 1. 安裝必要的依賴 如果你使用 webpack 作為構建工具&#xff0c;你可能需要安裝 css-loader 和 style-loader。如果你的項目使用 Create React App 或其他現代前端框架&#xff0c;這些可能已經內置了。 npm install --save-dev css-loade…

springboot結合baomidou dynamic-datasource組件實現多數據源

dynamic-datasource組件實現多數據源 一、背景介紹二、 思路方案三、過程四、總結五、升華 一、背景介紹 博主最近研發的項目中由于業務需要&#xff0c;在項目中使用到多個數據源。使用到了baomidou的dynamic-datasource組件來實現訪問不同的數據源。覺得挺有意思的也是進行了…

Redis事務(1)

什么是事務&#xff1f; Redis 的事務和 MySQL 的事務概念上是類似的. 都是把?系列操作綁定成?組. 讓這?組能夠批量執行。 但是注意體會 Redis 的事務和 MySQL 事務的區別: 弱化的原?性: redis 沒有 “回滾機制”. 只能做到這些操作 “批量執?”. 不能做到 “?個失敗就…

海外鏈游地鐵跑酷全自動搬磚掛機掘金變現項目,號稱單窗口一天收益30+(教程+工具)

一、項目概述 地鐵跑酷海外版國外版自動搬磚掛機掘金項目是一款結合了地鐵跑酷元素的在線游戲&#xff0c;為玩家提供一個全新的游戲體驗&#xff0c;使得玩家可以輕松地進行游戲&#xff0c;無需手動操作&#xff0c;節省時間和精力。 二、游戲特點 1. 自動化操作&#xff1…

AI應用案例:影像報告智能輔助編輯系統

今天給大家介紹一個醫療行業的案例“影像報告智能輔助編輯系統”&#xff01;該案例已經在某三甲醫院落地&#xff0c;模型準確度超過80%。 該項目上線后&#xff0c;保守估計&#xff0c;能為每位醫生的每一張報告至少省下1分鐘時間和2分鐘的精力&#xff0c;20位初級醫生&…

Django Web:搭建Websocket服務器(入門篇)

Django Web架構 搭建Websocket服務器&#xff08;1&#xff09; - 文章信息 - Author: 李俊才 (jcLee95) Visit me at CSDN: https://jclee95.blog.csdn.netMy WebSite&#xff1a;http://thispage.tech/Email: 291148484163.com. Shenzhen ChinaAddress of this article:htt…

如何在Windows 10上對硬盤進行碎片整理?這里提供步驟

隨著時間的推移&#xff0c;由于文件系統中的碎片&#xff0c;硬盤驅動器可能會開始以較低的效率運行。為了加快驅動器的速度&#xff0c;你可以使用內置工具在Windows 10中對其進行碎片整理和優化。方法如下。 什么是碎片整理 隨著時間的推移&#xff0c;組成文件的數據塊&a…