123 lines
3.4 KiB
Markdown
123 lines
3.4 KiB
Markdown
|
||
## 题目链接
|
||
https://leetcode-cn.com/problems/palindrome-linked-list/
|
||
|
||
## 思路
|
||
|
||
### 数组模拟
|
||
|
||
最直接的想法,就是把链表装成数组,然后再判断是否回文。
|
||
|
||
代码也比较简单。如下:
|
||
|
||
```
|
||
class Solution {
|
||
public:
|
||
bool isPalindrome(ListNode* head) {
|
||
vector<int> vec;
|
||
ListNode* cur = head;
|
||
while (cur) {
|
||
vec.push_back(cur->val);
|
||
cur = cur->next;
|
||
}
|
||
// 比较数组回文
|
||
for (int i = 0, j = vec.size() - 1; i < j; i++, j--) {
|
||
if (vec[i] != vec[j]) return false;
|
||
}
|
||
return true;
|
||
}
|
||
};
|
||
```
|
||
|
||
上面代码可以在优化,就是先求出链表长度,然后给定vector的初始长度,这样避免vector每次添加节点重新开辟空间
|
||
|
||
```
|
||
class Solution {
|
||
public:
|
||
bool isPalindrome(ListNode* head) {
|
||
|
||
ListNode* cur = head;
|
||
int length = 0;
|
||
while (cur) {
|
||
length++;
|
||
cur = cur->next;
|
||
}
|
||
vector<int> vec(length, 0); // 给定vector的初始长度,这样避免vector每次添加节点重新开辟空间
|
||
cur = head;
|
||
int index = 0;
|
||
while (cur) {
|
||
vec[index++] = cur->val;
|
||
cur = cur->next;
|
||
}
|
||
// 比较数组回文
|
||
for (int i = 0, j = vec.size() - 1; i < j; i++, j--) {
|
||
if (vec[i] != vec[j]) return false;
|
||
}
|
||
return true;
|
||
}
|
||
};
|
||
|
||
```
|
||
|
||
### 反转后半部分链表
|
||
|
||
分为如下几步:
|
||
|
||
* 用快慢指针,快指针有两步,慢指针走一步,快指针遇到终止位置时,慢指针就在链表中间位置
|
||
* 同时用pre记录慢指针指向节点的前一个节点,用来分割链表
|
||
* 将链表分为前后均等两部分,如果链表长度是奇数,那么后半部分多一个节点
|
||
* 将后半部分反转 ,得cur2,前半部分为cur1
|
||
* 按照cur1的长度,一次比较cur1和cur2的节点数值
|
||
|
||
如图所示:
|
||
|
||
<img src='../pics/234.回文链表.png' width=600> </img></div>
|
||
|
||
代码如下:
|
||
|
||
```
|
||
class Solution {
|
||
public:
|
||
bool isPalindrome(ListNode* head) {
|
||
if (head == nullptr || head->next == nullptr) return true;
|
||
ListNode* slow = head; // 慢指针,找到链表中间分位置,作为分割
|
||
ListNode* fast = head;
|
||
ListNode* pre = head; // 记录慢指针的前一个节点,用来分割链表
|
||
while (fast && fast->next) {
|
||
pre = slow;
|
||
slow = slow->next;
|
||
fast = fast->next->next;
|
||
}
|
||
pre->next = nullptr; // 分割链表
|
||
|
||
ListNode* cur1 = head; // 前半部分
|
||
ListNode* cur2 = reverseList(slow); // 反转后半部分,总链表长度如果是奇数,cur2比cur1多一个节点
|
||
|
||
// 开始两个链表的比较
|
||
while (cur1) {
|
||
if (cur1->val != cur2->val) return false;
|
||
cur1 = cur1->next;
|
||
cur2 = cur2->next;
|
||
}
|
||
return true;
|
||
}
|
||
// 反转链表
|
||
ListNode* reverseList(ListNode* head) {
|
||
ListNode* temp; // 保存cur的下一个节点
|
||
ListNode* cur = head;
|
||
ListNode* pre = nullptr;
|
||
while(cur) {
|
||
temp = cur->next; // 保存一下 cur的下一个节点,因为接下来要改变cur->next
|
||
cur->next = pre; // 翻转操作
|
||
// 更新pre 和 cur指针
|
||
pre = cur;
|
||
cur = temp;
|
||
}
|
||
return pre;
|
||
}
|
||
};
|
||
```
|
||
|
||
栈
|
||
|