判断回文链表.md 12.4 KB
Newer Older
1 2
# 如何高效判断回文链表

L
labuladong 已提交
3 4 5 6




L
labuladong 已提交
7 8


9 10
<p align='center'>
<a href="https://github.com/labuladong/fucking-algorithm" target="view_window"><img alt="GitHub" src="https://img.shields.io/github/stars/labuladong/fucking-algorithm?label=Stars&style=flat-square&logo=GitHub"></a>
L
labuladong 已提交
11
<a href="https://appktavsiei5995.pc.xiaoe-tech.com/index" target="_blank"><img class="my_header_icon" src="https://img.shields.io/static/v1?label=精品课程&message=查看&color=pink&style=flat"></a>
12 13 14 15
<a href="https://www.zhihu.com/people/labuladong"><img src="https://img.shields.io/badge/%E7%9F%A5%E4%B9%8E-@labuladong-000000.svg?style=flat-square&logo=Zhihu"></a>
<a href="https://space.bilibili.com/14089380"><img src="https://img.shields.io/badge/B站-@labuladong-000000.svg?style=flat-square&logo=Bilibili"></a>
</p>

L
labuladong 已提交
16
![](https://labuladong.github.io/algo/images/souyisou1.png)
17

L
labuladong 已提交
18
**通知:[数据结构精品课](https://aep.h5.xeknow.com/s/1XJHEO) 已更新到 V2.0,第 12 期刷题打卡 [开始报名](https://aep.xet.tech/s/XhcRc),点击这里体验 [刷题全家桶](https://labuladong.gitee.io/algo/images/others/%E5%85%A8%E5%AE%B6%E6%A1%B6.jpg)。**
19 20


L
labuladong 已提交
21 22 23 24 25 26 27

读完本文,你不仅学会了算法套路,还可以顺便解决如下题目:

| LeetCode | 力扣 | 难度 |
| :----: | :----: | :----: |
| [234. Palindrome Linked List](https://leetcode.com/problems/palindrome-linked-list/) | [234. 回文链表](https://leetcode.cn/problems/palindrome-linked-list/) | 🟢
| - | [剑指 Offer II 027. 回文链表](https://leetcode.cn/problems/aMhZSa/) | 🟢
28 29 30

**-----------**

L
labuladong 已提交
31
前文 [数组双指针技巧汇总](https://labuladong.github.io/article/fname.html?fname=双指针技巧)[子序列问题解题思路](https://labuladong.github.io/article/fname.html?fname=子序列问题模板) 讲解了回文串和回文序列相关的问题,先来简单回顾下。
L
labuladong 已提交
32 33 34

**寻找**回文串的核心思想是从中心向两端扩展:

L
labuladong 已提交
35 36 37
```java
// 在 s 中寻找以 s[left] 和 s[right] 为中心的最长回文串
String palindrome(String s, int left, int right) {
L
labuladong 已提交
38
    // 防止索引越界
L
labuladong 已提交
39 40 41 42 43
    while (left >= 0 && right < s.length()
            && s.charAt(left) == s.charAt(right)) {
        // 双指针,向两边展开
        left--;
        right++;
L
labuladong 已提交
44
    }
L
labuladong 已提交
45 46
    // 返回以 s[left] 和 s[right] 为中心的最长回文串
    return s.substring(left + 1, right);
L
labuladong 已提交
47 48 49
}
```

L
labuladong 已提交
50
因为回文串长度可能为奇数也可能是偶数,长度为奇数时只存在一个中心点,而长度为偶数时存在两个中心点,所以上面这个函数需要传入 `l``r`
L
labuladong 已提交
51

L
labuladong 已提交
52
**判断**一个字符串是不是回文串就简单很多,不需要考虑奇偶情况,只需要[双指针技巧](https://labuladong.github.io/article/fname.html?fname=双指针技巧),从两端向中间逼近即可:
L
labuladong 已提交
53

L
labuladong 已提交
54 55 56 57
```java
boolean isPalindrome(String s) {
    // 一左一右两个指针相向而行
    int left = 0, right = s.length() - 1;
L
labuladong 已提交
58
    while (left < right) {
L
labuladong 已提交
59
        if (s.charAt(left) != s.charAt(right)) {
L
labuladong 已提交
60
            return false;
L
labuladong 已提交
61 62 63
        }
        left++;
        right--;
L
labuladong 已提交
64 65 66 67 68 69 70 71 72 73 74
    }
    return true;
}
```

以上代码很好理解吧,**因为回文串是对称的,所以正着读和倒着读应该是一样的,这一特点是解决回文串问题的关键**

下面扩展这一最简单的情况,来解决:如何判断一个「单链表」是不是回文。

### 一、判断回文单链表

L
labuladong 已提交
75
看下力扣第 234 题「回文链表」:
L
labuladong 已提交
76

L
labuladong 已提交
77
输入一个单链表的头结点,判断这个链表中的数字是不是回文,函数签名如下:
L
labuladong 已提交
78

L
labuladong 已提交
79
```java
L
labuladong 已提交
80
boolean isPalindrome(ListNode head);
L
labuladong 已提交
81 82 83
```

比如说:
L
labuladong 已提交
84

L
labuladong 已提交
85
```
L
labuladong 已提交
86 87 88 89 90 91 92
输入: 1->2->null
输出: false

输入: 1->2->2->1->null
输出: true
```

L
labuladong 已提交
93 94 95
这道题的关键在于,单链表无法倒着遍历,无法使用双指针技巧。

那么最简单的办法就是,把原始链表反转存入一条新的链表,然后比较这两条链表是否相同。关于如何反转链表,可以参见前文 [递归翻转链表的一部分](https://labuladong.github.io/article/fname.html?fname=递归反转链表的一部分)
L
labuladong 已提交
96 97 98 99 100 101 102 103 104 105 106 107 108 109 110

其实,**借助二叉树后序遍历的思路,不需要显式反转原始链表也可以倒序遍历链表**,下面来具体聊聊。

对于二叉树的几种遍历方式,我们再熟悉不过了:

```java
void traverse(TreeNode root) {
    // 前序遍历代码
    traverse(root.left);
    // 中序遍历代码
    traverse(root.right);
    // 后序遍历代码
}
```

L
labuladong 已提交
111
[学习数据结构的框架思维](https://labuladong.github.io/article/fname.html?fname=学习数据结构和算法的高效方法) 中说过,链表兼具递归结构,树结构不过是链表的衍生。那么,**链表其实也可以有前序遍历和后序遍历**
L
labuladong 已提交
112 113 114 115 116 117 118 119 120

```java
void traverse(ListNode head) {
    // 前序遍历代码
    traverse(head.next);
    // 后序遍历代码
}
```

L
labuladong 已提交
121
这个框架有什么指导意义呢?如果我想正序打印链表中的 `val` 值,可以在前序遍历位置写代码;反之,如果想倒序遍历链表,就可以在后序遍历位置操作:
L
labuladong 已提交
122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153

```java
/* 倒序打印单链表中的元素值 */
void traverse(ListNode head) {
    if (head == null) return;
    traverse(head.next);
    // 后序遍历代码
    print(head.val);
}
```

说到这了,其实可以稍作修改,模仿双指针实现回文判断的功能:

```java
// 左侧指针
ListNode left;

boolean isPalindrome(ListNode head) {
    left = head;
    return traverse(head);
}

boolean traverse(ListNode right) {
    if (right == null) return true;
    boolean res = traverse(right.next);
    // 后序遍历代码
    res = res && (right.val == left.val);
    left = left.next;
    return res;
}
```

L
labuladong 已提交
154
这么做的核心逻辑是什么呢?**实际上就是把链表节点放入一个栈,然后再拿出来,这时候元素顺序就是反的**,只不过我们利用的是递归函数的堆栈而已,如下 GIF 所示:
L
labuladong 已提交
155

L
labuladong 已提交
156
![](https://labuladong.github.io/algo/images/回文链表/1.gif)
L
labuladong 已提交
157

L
labuladong 已提交
158
当然,无论造一条反转链表还是利用后序遍历,算法的时间和空间复杂度都是 O(N)。下面我们想想,能不能不用额外的空间,解决这个问题呢?
L
labuladong 已提交
159 160 161 162 163

### 二、优化空间复杂度

更好的思路是这样的:

L
labuladong 已提交
164
**1、先通过 [双指针技巧](https://labuladong.github.io/article/fname.html?fname=链表技巧) 中的快慢指针来找到链表的中点**
L
labuladong 已提交
165 166 167 168 169 170 171 172 173 174 175

```java
ListNode slow, fast;
slow = fast = head;
while (fast != null && fast.next != null) {
    slow = slow.next;
    fast = fast.next.next;
}
// slow 指针现在指向链表中点
```

L
labuladong 已提交
176
![](https://labuladong.github.io/algo/images/回文链表/1.jpg)
L
labuladong 已提交
177 178 179 180 181 182 183 184

**2、如果`fast`指针没有指向`null`,说明链表长度为奇数,`slow`还要再前进一步**

```java
if (fast != null)
    slow = slow.next;
```

L
labuladong 已提交
185
![](https://labuladong.github.io/algo/images/回文链表/2.jpg)
L
labuladong 已提交
186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201

**3、从`slow`开始反转后面的链表,现在就可以开始比较回文串了**

```java
ListNode left = head;
ListNode right = reverse(slow);

while (right != null) {
    if (left.val != right.val)
        return false;
    left = left.next;
    right = right.next;
}
return true;
```

L
labuladong 已提交
202
![](https://labuladong.github.io/algo/images/回文链表/3.jpg)
L
labuladong 已提交
203

L
labuladong 已提交
204
至此,把上面 3 段代码合在一起就高效地解决这个问题了,其中 `reverse` 函数很容易实现:
L
labuladong 已提交
205 206

```java
L
labuladong 已提交
207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229
boolean isPalindrome(ListNode head) {
    ListNode slow, fast;
    slow = fast = head;
    while (fast != null && fast.next != null) {
        slow = slow.next;
        fast = fast.next.next;
    }
    
    if (fast != null)
        slow = slow.next;
    
    ListNode left = head;
    ListNode right = reverse(slow);
    while (right != null) {
        if (left.val != right.val)
            return false;
        left = left.next;
        right = right.next;
    }
    
    return true;
}

L
labuladong 已提交
230 231 232 233 234 235 236 237 238 239 240 241
ListNode reverse(ListNode head) {
    ListNode pre = null, cur = head;
    while (cur != null) {
        ListNode next = cur.next;
        cur.next = pre;
        pre = cur;
        cur = next;
    }
    return pre;
}
```

L
labuladong 已提交
242 243 244
算法过程如下 GIF 所示:

![](https://labuladong.github.io/algo/images/kgroup/8.gif)
L
labuladong 已提交
245 246 247 248 249 250 251

算法总体的时间复杂度 O(N),空间复杂度 O(1),已经是最优的了。

我知道肯定有读者会问:这种解法虽然高效,但破坏了输入链表的原始结构,能不能避免这个瑕疵呢?

其实这个问题很好解决,关键在于得到`p, q`这两个指针位置:

L
labuladong 已提交
252
![](https://labuladong.github.io/algo/images/回文链表/4.jpg)
L
labuladong 已提交
253 254 255 256 257 258 259 260 261 262 263 264 265

这样,只要在函数 return 之前加一段代码即可恢复原先链表顺序:

```java
p.next = reverse(q);
```

篇幅所限,我就不写了,读者可以自己尝试一下。

### 三、最后总结

首先,寻找回文串是从中间向两端扩展,判断回文串是从两端向中间收缩。对于单链表,无法直接倒序遍历,可以造一条新的反转链表,可以利用链表的后序遍历,也可以用栈结构倒序处理单链表。

L
labuladong 已提交
266 267
具体到回文链表的判断问题,由于回文的特殊性,可以不完全反转链表,而是仅仅反转部分链表,将空间复杂度降到 O(1)。

L
labuladong 已提交
268 269
> 最后打个广告,我亲自制作了一门 [数据结构精品课](https://aep.h5.xeknow.com/s/1XJHEO),以视频课为主,手把手带你实现常用的数据结构及相关算法,旨在帮助算法基础较为薄弱的读者深入理解常用数据结构的底层原理,在算法学习中少走弯路。

270
**_____________**
L
labuladong 已提交
271

L
labuladong 已提交
272
**《labuladong 的算法小抄》已经出版,关注公众号查看详情;后台回复关键词「**进群**」可加入算法群;回复「**全家桶**」可下载配套 PDF 和刷题全家桶**
L
labuladong 已提交
273 274

![](https://labuladong.github.io/algo/images/souyisou2.png)
L
labuladong 已提交
275 276


277 278
======其他语言代码======

B
brucecat 已提交
279 280 281 282 283 284
[234.回文链表](https://leetcode-cn.com/problems/palindrome-linked-list)



### C++

285
```cpp
B
brucecat 已提交
286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 350 351 352 353 354 355 356 357
bool isPalindrome(ListNode* head) {
  if (head == nullptr || head->next == nullptr) //为空或者只有一个节点时,直接判断为true
    return true;
  ListNode* slow = head, * fast = head;
  while (fast != nullptr) {//首先找到中间节点
    slow = slow->next;
    fast = fast->next == nullptr? fast->next:fast->next->next; //因为链表长度可能是奇数或偶数,所以需要进行判断
  }

  ListNode* temp = nullptr,* pre = nullptr;//pre始终保持后续链表的头部,temp节点则作为中间零时替换的节点
  while (slow != nullptr) {//利用头插法,将当前节点与后续链表断链处理,反转后半部分的链表
    temp = slow->next;
    slow->next = pre;//建立连接
    pre = slow;//pre始终作为后续链表的头部
    slow = temp;
  }

  while (head !=nullptr && pre != nullptr) {//同步进行比较
    if (head->val != pre->val) {//值有不一样的,说明不是回文联表,直接返回false了
      return false;
    }
    head = head->next;//head向下走,直到走到空
    pre = pre->next;//pre节点也向下走,直到走到空
  }
  return true;//到此说明当前链表是回文链表返回true即可
}

```



### javascript

**后序遍历法**

```js
let left;
var isPalindrome = function(head) {
    left = head;
    return traverse(head);
};

var traverse= function(right){
    if(right == null) return true;
    let res = traverse(right.next);

    // 后序遍历
    res = res && (right.val === left.val);
    left = left.next;
    return res;
}
```



**双指针找到链表中点+链表翻转+链表对比**

```js
/**
 * Definition for singly-linked list.
 * function ListNode(val) {
 *     this.val = val;
 *     this.next = null;
 * }
 */
/**
 * @param {ListNode} head
 * @return {boolean}
 */

var isPalindrome = function (head) {
    if(head == null || head.next == null){
358 359 360
        return true;
    }

B
brucecat 已提交
361 362 363 364 365 366 367 368 369 370 371 372 373 374 375 376 377 378 379 380 381 382 383 384 385 386 387 388 389 390 391 392 393 394 395 396 397 398 399 400 401 402

    let left = head;
    let midNode = findMid(head);
    let right = reverse(midNode);

    // 开始比较
    while(right != null){
        if(left.val !== right.val){
            return false;
        }
        left = left.next;
        right = right.next;
    }

    return true;
};

// 双指针找到链表中间结点
var findMid = function (head){
    let fast, slow;
    fast = slow = head;
    while (fast != null && fast.next != null && slow != null) {
        slow = slow.next;
        fast = fast.next.next;
    }

    // slow 现在指向链表中点
    return slow;
}

// 翻转以head为头的链表
var reverse = function (head) {
    let pre = null, cur = head;
    while(cur!=null){
        let next = cur.next;
        cur.next = pre;
        pre = cur;
        cur = next;
    }
    return pre;
}

403
```
B
brucecat 已提交
404