链表中删除倒数第K个节点

问题描述

  分别实现两个函数,一个可以删除单链表中倒数第K个节点,另一个可以删除双链表中倒数第K个节点。

问题分析与解决

  从问题当中,我们只能得到一个链表和要删除的第K个节点的信息,于是就有以下思路:如果链表为空或者K<0时,直接返回;如若不然,遍历链表的每个节点,每经过一个节点K减1。比如对于1 --> 2 --> 3 --> 4该链表的过程如下:

  K = 5,所遍历的节点以及K值的变化:1 -- > 2 --> 3 --> 4  4,3,2,1;

  K = 4,所遍历的节点以及K值的变化:1 -- > 2 --> 3 --> 4  3,2,1,0;

  K = 3,所遍历的节点以及K值的变化:1 -- > 2 --> 3 --> 4  2,1,0,-1;

  K = 2,所遍历的节点以及K值的变化:1 -- > 2 --> 3 --> 4  1,0,-1,-2;

  由上可知,遍历链表中的节点,每经过一个节点K值减1的过程中,当K > 1时,说明链表中要删除的倒数第K个节点不存在,此时已经超出链表的长度;当K = 0时,此时正好要删除的是链表中的第一个节点,这是只需头节点指向头节点的下一个节点即可;那么对于K < 0时,该如何删除链表中倒数第K的节点呢?

  经过上面的步骤后,如果K<0,此时重新遍历链表,只不过这时是每经过一个节点K值增1。如下示例(K值保存经过减1后的结果):

  K = -1,所遍历的节点以及K值的变化:1  0;

  K = -2,所遍历的节点以及K值的变化:1 -- > 2   -1,0;

  在遍历链表K值增1的过程中,当K = 0时,所在的位置正好是要删除倒数第K个节点的前一个节点,此时只需将前一个节点指向要删除的节点的下一节点即可。

代码实现(单链表):

 1 class Node(object):
 2     def __init__(self, data):
 3         self.data = data
 4         self.next = None
 5 
 6 def createSingleLink():
 7     head = Node(1)
 8     cur = head
 9     for i in range(2, 10):
10         cur.next = Node(i)
11         cur = cur.next
12     return head
13 
14 def printSingleLink(head):
15     cur = head
16     while cur:
17         print(cur.data, end='')
18         if cur.next:
19             print('-->', end='')
20         cur = cur.next
21 
22 def removeLastKthNode(head, lastKth):
23     if head is None or lastKth < 0:
24         return head
25     cur = head
26     # lastKth -= 1
27     while cur:
28         lastKth -= 1
29         cur = cur.next
30     # print(lastKth)
31     if lastKth == 0:
32         head = head.next
33     if lastKth < 0:
34         cur = head
35         lastKth += 1
36         while lastKth < 0:
37             cur = cur.next
38             lastKth += 1
39         cur.next = cur.next.next
40     return head
41 
42 if __name__ == '__main__':
43     singleHead = createSingleLink()
44     printSingleLink(singleHead)
45     print()
46     newSingleHead = removeLastKthNode(singleHead, 6)
47     printSingleLink(newSingleHead)
View Code

  上述代码是对删除单链表倒数第K个节点的实现,那么对于双链表的实现过程和单链表的过程相同,只不过在删除某一节时要注意节点的前驱指针的指向。

代码实现(双链表):

 1 class Node(object):
 2     def __init__(self, data):
 3         self.last = None
 4         self.data = data
 5         self.next = None
 6 
 7 def createDoubleLink():
 8     head = Node(1)
 9     cur = head
10     for i in range(2, 10):
11         cur.next = Node(i)
12         cur = cur.next
13     return head
14 
15 def printLink(head):
16     cur = head
17     while cur:
18         print(cur.data, end='')
19         if cur.next:
20             print('-->', end='')
21         cur = cur.next
22 
23 def removeLastKthNode(head, lastKth):
24     if head is None or lastKth < 0:
25         return head
26     cur = head
27     while cur:
28         lastKth -= 1
29         cur = cur.next
30     if lastKth == 0:
31         head = head.next
32         head.last = None
33     if lastKth < 0:
34         cur = head
35         lastKth += 1
36         while lastKth < 0:
37             cur = cur.next
38             lastKth += 1
39         if cur.next.next:
40             cur.next = cur.next.next
41             cur.next.last = cur
42         else:
43             cur.next = None
44     return head
45 
46 if __name__ == '__main__':
47     doubleLinkHead = createDoubleLink()
48     printLink(doubleLinkHead)
49     print()
50     newDoubleLinkHead = removeLastKthNode(doubleLinkHead, 6)
51     printLink(newDoubleLinkHead)
View Code

 

posted @ 2019-10-20 11:19  Dabric  阅读(1251)  评论(0编辑  收藏  举报
TOP