hot 100 第二十四题 24.回文链表

题目:

给你一个单链表的头节点 head ,请你判断该链表是否为回文链表。如果是,返回 true ;否则,返回 false

示例 1:

复制代码
输入:head = [1,2,2,1]
输出:true

示例 2:

复制代码
输入:head = [1,2]
输出:false

核心思路

快慢指针 + 反转后半部分 + 比较

步骤:

  1. 用快慢指针找到链表中点

  2. 反转后半部分链表

  3. 比较前半部分和反转后的后半部分

  4. (可选)恢复链表

    原链表: 1 → 2 → 2 → 1

    步骤1: 找中点
    1 → 2 → 2 → 1

    中点

    步骤2: 反转后半部分
    前半: 1 → 2
    后半: 1 → 2 (反转后)

    步骤3: 比较
    1 == 1 ✓
    2 == 2 ✓
    是回文

题解:

java 复制代码
class Solution {
    public boolean isPalindrome(ListNode head) {
        if (head == null || head.next == null) return true;
        
        // 1. 用快慢指针找到中点
        ListNode slow = head;
        ListNode fast = head;
        while (fast != null && fast.next != null) {
            slow = slow.next;
            fast = fast.next.next;
        }
        
        // 2. 反转后半部分
        ListNode secondHalf = reverseList(slow);
        
        // 3. 比较前半部分和后半部分
        ListNode p1 = head;
        ListNode p2 = secondHalf;
        while (p2 != null) {  // 后半部分较短或相等
            if (p1.val != p2.val) {
                return false;
            }
            p1 = p1.next;
            p2 = p2.next;
        }
        
        return true;
    }
    
    // 反转链表
    private ListNode reverseList(ListNode head) {
        ListNode prev = null;
        ListNode curr = head;
        while (curr != null) {
            ListNode next = curr.next;
            curr.next = prev;
            prev = curr;
            curr = next;
        }
        return prev;
    }
}
```

### 详细演示(奇数节点)
```
链表: 1 → 2 → 3 → 2 → 1

步骤1: 找中点
------------------
slow 和 fast 都从 head 开始

第1次移动:
slow: 1 → 2
fast: 1 → 2 → 3

第2次移动:
slow: 2 → 3
fast: 3 → 2 → 1

第3次移动:
slow: 3
fast: 1 → null (fast.next = null,停止)

slow 指向中点 3

步骤2: 反转后半部分(从3开始)
------------------
原链表: 1 → 2 → 3 → 2 → 1

反转 3 → 2 → 1:
null ← 3 ← 2 ← 1

现在:
前半: 1 → 2 → 3
后半: 1 → 2 ← 3 (secondHalf指向1)

步骤3: 比较
------------------
p1 从 head (1) 开始
p2 从 secondHalf (1) 开始

比较:
p1=1, p2=1 ✓
p1=2, p2=2 ✓
p1=3, p2=3 ✓
p2 走到头,返回 true
```

### 详细演示(偶数节点)
```
链表: 1 → 2 → 2 → 1

步骤1: 找中点
------------------
第1次移动:
slow: 1 → 2
fast: 1 → 2 → 2

第2次移动:
slow: 2 (第二个2)
fast: 2 → 1 → null

fast.next = null,停止
slow 指向后半部分的起点

步骤2: 反转后半部分(从第二个2开始)
------------------
原链表: 1 → 2 → 2 → 1

反转 2 → 1:
null ← 2 ← 1

现在:
前半: 1 → 2
后半: 1 → 2 (secondHalf指向1)

步骤3: 比较
------------------
p1=1, p2=1 ✓
p1=2, p2=2 ✓
p2 走到头,返回 true

本质

回文链表问题的核心技巧:

  1. 快慢指针 --- 找到中点,O(1) 空间
  2. 反转链表 --- 使后半部分可以从前往后遍历
  3. 双指针比较 --- 同时遍历两部分
相关推荐
CSharp精选营1 小时前
关系型 vs 非关系型:从原理到选型,一文搞定数据库核心分类
数据结构·nosql·关系型数据库·非关系型数据库·技术选型
刘马想放假3 天前
Modbus 全栈技术解析:TCP、RTU、ASCII、RTU over TCP
数据结构·网络协议
北域码匠4 天前
冒泡排序太慢?鸡尾酒排序双向优化,原生 C# 零第三方库完整代码
数据结构·排序算法·泛型·c# 算法·鸡尾酒排序·原生 c# 开发·冒泡排序优化·嵌入式算法
Darling噜啦啦11 天前
列表转树算法深度解析:从 Map 到 Reduce 的两种实现,面试高频考点
数据结构·算法·面试
小小工匠12 天前
Redis - 事务机制:能实现 ACID 属性吗
数据结构·redis·性能优化·并发·持久化
玖玥拾12 天前
C/C++ 数据结构(七)栈、容器适配器
c语言·数据结构·c++··容器适配器
Qres82112 天前
算法复键——树状数组
数据结构·算法
牛油果子哥q12 天前
并查集(DSU)超精讲,路径压缩、按秩合并、万能模板、连通性判定、最小生成树与刷题实战全解
数据结构·c++·最小生成树·并查集
凌波粒12 天前
LeetCode--491.递增子序列(回溯算法)
数据结构·算法·leetcode
疯狂成瘾者12 天前
Java 集合 LinkedList 详解:链表结构、常用方法和队列使用
java·开发语言·链表