描述
将一个节点数为 size 链表 m 位置到 n 位置之间的区间反转,要求时间复杂度 𝑂(𝑛)O(n),空间复杂度 𝑂(1)O(1)。
例如:
给出的链表为 1→2→3→4→5→𝑁𝑈𝐿𝐿1→2→3→4→5→NULL, 𝑚=2,𝑛=4m=2,n=4,
返回 1→4→3→2→5→𝑁𝑈𝐿𝐿1→4→3→2→5→NULL.
数据范围: 链表长度 0<𝑠𝑖𝑧𝑒≤10000<size≤1000,0<𝑚≤𝑛≤𝑠𝑖𝑧𝑒0<m≤n≤size,链表中每个节点的值满足 ∣𝑣𝑎𝑙∣≤1000∣val∣≤1000
要求:时间复杂度 𝑂(𝑛)O(n) ,空间复杂度 𝑂(𝑛)O(n)
进阶:时
/**
* struct ListNode {
* int val;
* struct ListNode *next;
* ListNode(int x) : val(x), next(nullptr) {}
* };
*/
class Solution {
public:
/**
* 代码中的类名、方法名、参数名已经指定,请勿修改,直接返回方法规定的值即可
*
*
* @param head ListNode类
* @param m int整型
* @param n int整型
* @return ListNode类
*/
ListNode* reverseBetween(ListNode* head, int m, int n) {
// write code here
ListNode* res = new ListNode(-1);
res->next = head;
ListNode* pre = res;
ListNode* cur = head;
for(int i = 1; i<m ;i++)
{
pre = cur;
cur = cur->next;
}
for(int i = m; i<n; i++)
{
ListNode* temp = cur->next;
cur->next = temp->next;
temp->next = pre->next;
pre->next = temp;
}
return res->next;
}
};
让我们详细解释一下为什么 return res->next;
可以实现去掉哑节点并返回修改后的链表的头部:
-
创建哑节点
ListNode* res = new ListNode(-1);
res->next = head;
这里创建了一个新的
ListNode
实例res
,其val
字段被设置为-1
(但这并不重要,因为通常我们不会使用它的值)。重要的是,我们将res
的next
指针指向了原始链表的头部head
。 -
找到要反转的区间的起始和结束位置:
通过两个循环,你找到了要反转的区间的起始节点(pre
指向它前一个节点)和结束节点(cur
指向它)。 -
反转区间:
在第二个循环中,你反转了从m
到n
(不包括n
)之间的链表部分。 -
返回结果:
return res->next;
在反转完成后,你返回了
res->next
。此时,res->next
指向的是原始链表的头部(或反转后链表的头部,如果m
是1),因为res
仅仅是一个指向链表的指针,它的next
字段指向链表的第一个实际节点。
为什么可以去掉哑节点?
- 由于
res
只是一个指向链表的额外指针,而不是链表的一部分,因此它不会在反转过程中被修改。这意味着res->next
始终指向链表的第一个实际节点。 - 在返回结果时,你返回的是
res->next
而不是res
。这样,调用者得到的是一个指向链表的指针,而不是指向哑节点的指针。 - 哑节点
res
本身在函数返回后不再需要,因为它只是一个临时的辅助节点。但由于它是在堆上分配的(使用new
),如果你真的关心内存管理,你可能需要在某个地方释放它(但在这个特定的情境下,通常不会释放它,因为它是一个临时的、辅助的节点,其生命周期与整个程序或特定数据结构相同)。
总之,return res->next;
之所以可以实现去掉哑节点,是因为你只返回了哑节点所指向的链表的头部,而没有返回哑节点本身。