LeetCodeAnimation/notes/LeetCode第19号问题:删除链表的倒数第N个节点.md
2019-05-02 16:23:13 +08:00

78 lines
2.2 KiB
Java
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

# LeetCode 19 号问题删除链表的倒数第 N 个节点
> 本文首发于公众号五分钟学算法[图解 LeetCode ](<https://github.com/MisterBooo/LeetCodeAnimation>)系列文章之一。
>
> 个人网站[https://www.cxyxiaowu.com](https://www.cxyxiaowu.com)
题目来源于 LeetCode 上第 19 号问题删除链表的倒数第 N 个节点题目难度为 Medium目前通过率为 34.4%
### 题目描述
给定一个链表删除链表的倒数第 *n* 个节点并且返回链表的头结点
**示例**
```
给定一个链表: 1->2->3->4->5, n = 2.
当删除了倒数第二个节点后链表变为 1->2->3->5.
```
**说明**
给定的 *n* 保证是有效的
**进阶**
你能尝试使用一趟扫描实现吗
### 题目解析
采取双重遍历肯定是可以解决问题的但题目要求我们一次遍历解决问题那我们的思路得发散一下
我们可以设想假设设定了双指针`p``q`的话`q`指向末尾的`NULL``p``q`之间相隔的元素个数为`n`那么删除掉`p`的下一个指针就完成了要求
- 设置虚拟节点`dummyHead`指向`head`
- 设定双指针`p``q`初始都指向虚拟节点`dummyHead`
- 移动`q`直到`p``q`之间相隔的元素个数为`n`
- 同时移动`p``q`直到`q`指向的为`NULL`
- `p`的下一个节点指向下下个节点
### 动画描述
![](https://bucket-1257126549.cos.ap-guangzhou.myqcloud.com/20181106162853.gif)
### 代码实现
```
class Solution {
public:
ListNode* removeNthFromEnd(ListNode* head, int n) {
ListNode* dummyHead = new ListNode(0);
dummyHead->next = head;
ListNode* p = dummyHead;
ListNode* q = dummyHead;
for( int i = 0 ; i < n + 1 ; i ++ ){
q = q->next;
}
while(q){
p = p->next;
q = q->next;
}
ListNode* delNode = p->next;
p->next = delNode->next;
delete delNode;
ListNode* retNode = dummyHead->next;
delete dummyHead;
return retNode;
}
};
```
![](https://bucket-1257126549.cos.ap-guangzhou.myqcloud.com/blog/fz0rq.png)