1.题目描述
2.解题思路
方法一:顺序查找
最简单直接的方法即为顺序查找,假设当前链表的长度为 n,则我们知道链表的倒数第 k 个节点即为正数第 n−k 个节点,此时我们只需要顺序遍历到链表的第 n−k 个节点即为倒数第 k 个节点。
我们首先求出链表的长度 n,然后顺序遍历到链表的第 n−k 个节点返回即可。
复杂度分析
时间复杂度:O(n),其中 n 为链表的长度。需要两次遍历。
空间复杂度:O(1)。
方法二:双指针
快慢指针的思想。我们将第一个指针fast 指向链表的第 k+1 个节点,第二个指针 slow 指向链表的第一个节点,此时指针 fast 与 slow 二者之间刚好间隔 k 个节点。此时两个指针同步向后走,当第一个指针 fast 走到链表的尾部空节点时,则此时 slow 指针刚好指向链表的倒数第k个节点。
我们首先将fast 指向链表的头节点,然后向后走 k 步,则此时 fast 指针刚好指向链表的第 k+1 个节点。
我们首先将slow 指向链表的头节点,同时slow 与 fast 同步向后走,当fast 指针指向链表的尾部空节点时,则此时返回 slow 所指向的节点即可。
复杂度分析
时间复杂度:O(n),其中 n 为链表的长度。我们使用快慢指针,只需要一次遍历即可,复杂度为 O(n)。
空间复杂度:O(1)。
3.代码实现
方法一:
class Solution {
public ListNode getKthFromEnd(ListNode head, int k) {
int n = 0;
ListNode node = null;
for (node = head; node != null; node = node.next) {
n++;
}
for (node = head; n > k; n--) {
node = node.next;
}
return node;
}
}
方法二:
class Solution {
public ListNode getKthFromEnd(ListNode head, int k) {
ListNode fast = head;
ListNode slow = head;
while (fast != null && k > 0) {
fast = fast.next;
k--;
}
while (fast != null) {
fast = fast.next;
slow = slow.next;
}
return slow;
}
}