算法(TS):回文链表

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

示例 1:

输入:head = [1,2,2,1] 输出:true

示例 2:

输入:head = [1,2] 输出:false


提示:

  • 链表中节点数目在范围[1, 105] 内
  • 0 <= Node.val <= 9

进阶: 你能否用 O(n) 时间复杂度和 O(1) 空间复杂度解决此题?

解答一

将链表中节点保存到一个Map中,key 为节点的次序,value 为节点。对比 Map 中第 n 个节点与第 total - n -1 个节点的值是否相等。

ts 复制代码
/**
 * Definition for singly-linked list.
 * class ListNode {
 *     val: number
 *     next: ListNode | null
 *     constructor(val?: number, next?: ListNode | null) {
 *         this.val = (val===undefined ? 0 : val)
 *         this.next = (next===undefined ? null : next)
 *     }
 * }
 */

function isPalindrome(head: ListNode | null): boolean {
    if (!head) return false
    const nodeMap = new Map<number,ListNode>()
    let count = 0
    while(head) {
        nodeMap.set(count,head)
        head = head.next
        count++
    }

    const total = count
    count = 0
    const middle = Math.floor(total/2)
    while(count < middle) {
        if (nodeMap.get(count).val !== nodeMap.get(total-count-1).val) {
            return false
        }
        count++
    }

    return true
};

**

**

时间复杂度O(n + n/2),空间复杂度O(n)

解答二

  1. 定义两个指针,fast 和 low,fast 的移动速度是 low 的两倍,当fast移动到链表末尾时,low就移动到了链表中间
  2. 反转 low 到链表末尾这部分节点
  3. 将 fast 重置到链表头,同步移动fast和low,比较他们指向的节点值是否相等,不相同则不是回文链表
ts 复制代码
/**
 * Definition for singly-linked list.
 * class ListNode {
 *     val: number
 *     next: ListNode | null
 *     constructor(val?: number, next?: ListNode | null) {
 *         this.val = (val===undefined ? 0 : val)
 *         this.next = (next===undefined ? null : next)
 *     }
 * }
 */

function isPalindrome(head: ListNode | null): boolean {
    if (!head) return false
    let fast = head
    let low = head
    while(fast && fast.next) {
        fast = fast.next.next
        low = low.next
    }
    // 如果 fast 不为 null,则说明传入的链表节点数为奇数
    if (fast) {
        low = low.next
    }

    low = reserve(low)
    fast = head
    while(low) {
        if (fast.val !== low.val) {
            return false
        }
        low = low.next
        fast = fast.next
    }

    return true
};

function reserve(head: ListNode):ListNode {
    let prev = null
    let current = null
    while(head) {
        prev = current
        current = head
        head = head.next
        current.next = prev
    }

    return current
}

时间复杂度O(n),空间复杂度O(1)

相关推荐
董董灿是个攻城狮3 小时前
5分钟搞懂什么是窗口注意力?
算法
Dann Hiroaki3 小时前
笔记分享: 哈尔滨工业大学CS31002编译原理——02. 语法分析
笔记·算法
qqxhb5 小时前
零基础数据结构与算法——第四章:基础算法-排序(上)
java·数据结构·算法·冒泡·插入·选择
FirstFrost --sy6 小时前
数据结构之二叉树
c语言·数据结构·c++·算法·链表·深度优先·广度优先
森焱森6 小时前
垂起固定翼无人机介绍
c语言·单片机·算法·架构·无人机
搂鱼1145147 小时前
(倍增)洛谷 P1613 跑路/P4155 国旗计划
算法
Yingye Zhu(HPXXZYY)7 小时前
Codeforces 2021 C Those Who Are With Us
数据结构·c++·算法
无聊的小坏坏8 小时前
三种方法详解最长回文子串问题
c++·算法·回文串
长路 ㅤ   8 小时前
Java后端技术博客汇总文档
分布式·算法·技术分享·编程学习·java后端
秋说8 小时前
【PTA数据结构 | C语言版】两枚硬币
c语言·数据结构·算法