视频1 视频21 视频41 视频61 视频文章1 视频文章21 视频文章41 视频文章61 推荐1 推荐3 推荐5 推荐7 推荐9 推荐11 推荐13 推荐15 推荐17 推荐19 推荐21 推荐23 推荐25 推荐27 推荐29 推荐31 推荐33 推荐35 推荐37 推荐39 推荐41 推荐43 推荐45 推荐47 推荐49 关键词1 关键词101 关键词201 关键词301 关键词401 关键词501 关键词601 关键词701 关键词801 关键词901 关键词1001 关键词1101 关键词1201 关键词1301 关键词1401 关键词1501 关键词1601 关键词1701 关键词1801 关键词1901 视频扩展1 视频扩展6 视频扩展11 视频扩展16 文章1 文章201 文章401 文章601 文章801 文章1001 资讯1 资讯501 资讯1001 资讯1501 标签1 标签501 标签1001 关键词1 关键词501 关键词1001 关键词1501 专题2001
Python数据结构与算法之链表定义的使用详解
2020-11-27 14:22:46 责编:小采
文档


这篇文章主要介绍了Python数据结构与算法之链表定义与用法,结合具体实例形式较为详细的分析了单链表、循环链表等的定义、使用方法与相关注意事项,需要的朋友可以参考下

本文实例讲述了Python数据结构与算法之链表定义与用法。分享给大家供大家参考,具体如下:

本文将为大家讲解:

(1)从链表节点的定义开始,以类的方式,面向对象的思想进行链表的设计

(2)链表类插入和删除等成员函数实现时需要考虑的边界条件,
prepend(头部插入)、pop(头部删除)、append(尾部插入)、pop_last(尾部删除)

2.1 插入:

空链表
链表长度为1
插入到末尾

2.2 删除

空链表
链表长度为1
删除末尾元素

(3)从单链表到单链表的一众变体:

带尾节点的单链表
循环单链表
双链表

1. 链表节点的定义

class LNode:
 def __init__(self, elem, next_=None):
 self.elem = elem
 self.next = next_

2. 单链表的实现

重点理解插入、删除的实现及其需要考虑的边界条件:

class LinkedListUnderflow(ValueError):
 pass
class LList:
 def __init__(self):
 self._head = None
 def is_empty(self):
 return self._head is None
 def prepend(self, elem):
 self._head = LNode(elem, self._head)
 def pop(self):
 if self._head is None:
 raise LinkedListUnderflow('in pop')
 e = self._head.elem
 self._head = self._head.next
 return e
 def append(self, elem):
 if self._head is None:
 self._head = LNode(elem)
 return
 p = self._head
 while p.next is not None:
 p = p.next
 p.next = LNode(elem)
 def pop_last(self):
 if self._head is None:
 raise LinkedListUnderflow('in pop_last')
 p = self._head
 if p.next is None:
 e = p.elem
 self._head = None
 return e
 while p.next.next is not None:
 p = p.next
 e = p.next.elem
 p.next = None
 return e

简单总结:

(0)能够访问 p.next.next 的前提是 p.next 不为空;
(1)尾部插入,如果链表不为空,需且仅需改变的是尾部节点的指针;
(2)尾部删除,如果链表长度不为空,需且仅需改变的是倒数第二个节点的指针。

单链表的简单变形:具有尾部节点的单链表

class LList1(LList):
 def __init__(self):
 LList.__init__(self)
 self._rear = None
 ...

我们仅需重写的是:头部的插入、尾部的插入、尾部的删除

def prepend(self, elem):
 if self._head is None:
 self._head = LNode(elem)
 self._rear = self._head
 else:
 self._head = LNode(elem, self._head)
def append(self, elem):
 if self._head is None:
 self._head = LNode(elem)
 self._rear = self._head
 else:
 self._rear.next = LNode(elem)
 self._rear = self._rear.next
def pop_last(self):
 if self._head is None:
 raise LinkedListUnderflow('in pop_last')
 p = self._head
 if p.next is None:
 e = p.elem
 self._head = None
 return e
 while p.next.next is not None:
 p = p.next
 e = p.next.elem
 self._rear = p
 p.next = None
 return e

单链表的变体:循环单链表

class LCList:
 def __init__(self):
 self._rear = None
 def prepend(self, elem):
 if self._rear is None:
 self._rear = LNode(elem)
 self._rear.next = self._rear
 else:
 self._rear.next = LNode(elem, self._rear.next)
 def append(self, elem):
 self.prepend(elem)
 self_rear = self._rear.next
 def pop(self):
 if self._rear is None:
 raise LinkedListUnderflow('in pop')
 p = self._rear.next
 if p is None:
 self._rear = None
 else:
 self._rear.next = p.next
 return p.elem
 def printall(self):
 if self._rear is None:
 raise ...
 p = self._rear.next
 while True:
 print(p.elem)
 if p is self._rear:
 break
 p = p.next

下载本文
显示全文
专题