输入一个链表,输出该链表中倒数第k个结点。为了符合大多数人的习惯,本题从1开始计数,即链表的尾节点是倒数第一个节点。例如,一个链表有6个节点。这链表的倒数第三个节点是值为4的节点
链表编程能力,代码的鲁棒性。考虑到特殊情况
此题与LeetCode第19题类似,但比19题情形要更复杂一点。
方法一:直接法, 两次遍历
1.第一次遍历求出链表长度。
2.第二次遍历返回指定结点。
3.注意特殊情况。
方法二:双指针法。
我们可以定义两个指针。第一个指针从链表的头指针开始遍历向前走k-1,第二个指针保持不动;从第k步开始,第二个指针也开始从链表的头指针开始遍历。由于两个指针的距离保持在k-1,当第一个(走在前面的)指针到达链表的尾结点时,第二个指针(走在后面的)指针正好是倒数第k个结点。
效果示意图,以链表总共6个结点,求倒数第3个结点为例:
/*
struct ListNode {
int val;
struct ListNode *next;
ListNode(int x) :
val(x), next(NULL) {
}
};*/
class Solution {
public:
ListNode* FindKthToTail(ListNode* head, unsigned int k) {
if (head==NULL || k<=0)
return NULL;
int n = 0;
for (ListNode* temp = head;temp!=NULL;temp=temp->next)
{
n++;
}
if (n<k) return NULL;
ListNode* p = head;
for (int i=0;i<n-k;i++)
{
p = p->next;
}
return p;
}
};
/*
struct ListNode {
int val;
struct ListNode *next;
ListNode(int x) :
val(x), next(NULL) {
}
};*/
class Solution {
public:
ListNode* FindKthToTail(ListNode* head, unsigned int k) {
if (head==NULL || k<=0)
return NULL;
ListNode* p = head;
ListNode* q = head;
// 从第1一个节点开始计数,所以走k-1步到达k节点,注意判断k是否超过链表总长度
for (int i=0;i<k-1;i++)
{
if (q->next!=NULL)
q = q->next;
else
return NULL;
}
while(q->next!=NULL)
{
p = p->next;
q = q->next;
}
return p;
}
};
# -*- coding:utf-8 -*-
# class ListNode:
# def __init__(self, x):
# self.val = x
# self.next = None
class Solution:
def FindKthToTail(self, head, k):
# write code here
if head==None or k<=0:
return None
temp = head
n = 0
while temp:
n+=1
temp = temp.next
if n<k:
return None
p = head
for i in range(n-k):
p = p.next
return p
# -*- coding:utf-8 -*-
# class ListNode:
# def __init__(self, x):
# self.val = x
# self.next = None
class Solution:
def FindKthToTail(self, head, k):
# write code here
if head==None or k<=0:
return None
p = head
q = head
for i in range(k-1):
if q.next!=None:
q = q.next
else:
return None
while q.next!=None:
p = p.next
q = q.next
return p