LeetCode 234:回文链表

题目描述

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

示例 1:

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

示例 2:

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

解题思路

1.栈+双指针

大致思路:使用双指针定位中心元素并将链表前半部分入栈,挨个出栈并遍历后半部分做比较,如果不想等则为false;

java 复制代码
class Solution {
    public boolean isPalindrome(ListNode head) {
        if (head == null || head.next == null) {
            return true;
        }
        
        // 使用快慢指针找到链表中点
        ListNode slow = head;
        ListNode fast = head;
        
        LinkedListStack<Integer> stack = new LinkedListStack<>();
        
        // 将前半部分入栈
        while (fast != null && fast.next != null) {
            stack.push(slow.val);
            slow = slow.next;
            fast = fast.next.next;
        }
        
        // 如果链表长度是奇数,跳过中间节点
        if (fast != null) {
            slow = slow.next;
        }
        
        // 比较后半部分和栈中元素
        while (slow != null) {
            if (stack.pop() != slow.val) {
                return false;
            }
            slow = slow.next;
        }
        
        return true;
    }
}

public class LinkedListStack<T> {
    // 定义链表节点
    private static class Node<T> {
        T data;
        Node<T> next;
        
        Node(T data) {
            this.data = data;
            this.next = null;
        }
    }
    
    private Node<T> top; // 栈顶节点
    private int size;    // 栈的大小
    
    public LinkedListStack() {
        top = null;
        size = 0;
    }
    
    // 入栈操作
    public void push(T item) {
        Node<T> newNode = new Node<>(item);
        newNode.next = top; // 新节点的next指向原来的栈顶
        top = newNode;     // 更新栈顶为新节点
        size++;
    }
    
    // 出栈操作
    public T pop() {
        if (isEmpty()) {
            throw new IllegalStateException("Stack is empty");
        }
        T item = top.data;  // 获取栈顶数据
        top = top.next;     // 栈顶指向下一个节点
        size--;
        return item;
    }
    
    // 判断栈是否为空
    public boolean isEmpty() {
        return top == null;
    }
}

2.反转链表+双指针

java 复制代码
class Solution {
    public boolean isPalindrome(ListNode head) {
        if (head == null || head.next == null) return true;
        
        // 1. 快慢指针找中点
        ListNode slow = head, fast = head;
        while (fast != null && fast.next != null) {
            slow = slow.next;
            fast = fast.next.next;
        }
        
        // 2. 反转后半部分
        ListNode reversedSecondHalf = reverse(slow);
        
        // 3. 比较前后两部分
        ListNode p1 = head, p2 = reversedSecondHalf;
        boolean isPalindrome = true;
        while (p2 != null) {
            if (p1.val != p2.val) {
                isPalindrome = false;
                break;
            }
            p1 = p1.next;
            p2 = p2.next;
        }
        
        return isPalindrome;
    }
    
    private ListNode reverse(ListNode head) {
        //prev指向反转后链表的最后一个 初始为null  curr指向要反转的那个节点,从head开始
        ListNode prev = null, curr = head;
        while (curr != null) {
            //将反转的下一个节点先记录下来
            ListNode next = curr.next;
            //使用头插法
            curr.next = prev;
            //将当前反转的节点设置为反转后节点的最后一个
            prev = curr;
            //反转下一个节点
            curr = next;
        }
        return prev;
    }
}
相关推荐
CoderYanger4 小时前
14届蓝桥杯省赛Java A 组Q1~Q3
java·开发语言·线性代数·算法·职场和发展·蓝桥杯
会编程的土豆6 小时前
【日常做题】 代码随想录(岛屿最大面积+寻宝)
数据结构·算法·图论
阿洛学长6 小时前
汉洛塔结构思维
算法
木子n16 小时前
第2篇:坐标变换与数学基础:FOC算法的核心数学工具
算法·电机控制·foc
阿Y加油吧6 小时前
两道经典 DP 题:零钱兑换 & 单词拆分(完全背包 + 字符串 DP)
算法
疯狂打码的少年6 小时前
有序线性表删除一个元素:顺序存储 vs 单链表,平均要移动多少个元素?
数据结构·算法·链表
y = xⁿ6 小时前
20天速通LeetCode day07:前缀和
数据结构·算法·leetcode
小雅痞7 小时前
[Java][Leetcode hard] 42. 接雨水
java·开发语言·leetcode
载数而行5207 小时前
算法集训1:模拟,枚举,错误分析,前缀和,差分
算法
hehelm7 小时前
vector模拟实现
前端·javascript·算法