线性表是由 n ( n > = 0 ) n(n>=0) n(n>=0)个数据元素(节点) a 1 、 a 2 、 a 3 、 … 、 a n a1、a2、a3、…、an a1、a2、a3、…、an 成的有限序列。该序列中的所有节点都具有相同的数据类型。其中,数据元素的个数 n n n 称为线性表的长度。
当n=0时,称为空表。当n>0时,为非空的线性表,记作(a1,a2,…,an)。
线性表的主要存储结构:
定义:把线性表的节点按逻辑顺序依次存放在一组地址连续的存储单元中,用这种方法存储的线性表称为顺序表,即顺序存储的线性表。
特点:
初始化顺序表时先给顺序表分配内存空间,声明数组的最大容量。Python中实例对象初始化一般会用到 __init__()方法。代码实现如下:
def __init__(self, max):
'''
初始化
:paramma:顺序表最大容量
'''
sel.max = max
sel.index = 0
self.data = [None for_ in range(self.max)]
通过下标值来查找元素,如果下标值合法,则返回该下标值上元素,否则抛出异常,代码实现如下。
def __getitem__(self, index):
'''
获取下标值为index的元素
:param index:下标值
:return : 下标值为 index 的元素
'''
if index < 0 or index >= self.index:
raise IndexError('index非法')
else:
return self.data[index]
修改下标值为 index 的元素的值,如果 index 非法,则抛出异常;否则修改数据元素的值,代码实现如下。
def__setitem__(self,index,value):
'''
修改下标值为index的元素的值
:paramindex:下标值
:paramvalue:待替换的元素
:return:
'''
if index < 0 or index >= self.index:
raiseIndexError('index非法')
else:
self.data[index] = value
若顺序表为空,则 self.index 为0,所以只需要判断 self.index 是否为 0 即可,代码实现如下:
def empty(self):
'''
判断顺序表是否为空
:return:TrueorFalse
'''
return self.index is 0
插入元素前需要先判断顺序表是否已经达到最大容量,如果达到最大容量,则退出程序;否则在顺序表尾部插入元素,代码实现如下:
def append(self, value):
'''
表尾插入元素
:paramvalue:待插入的元素
:return:顺序表已满的出口
'''
if self.index is self.max:
return
else:
self.data[self.index] = value
self.index += 1
先判断 index 下标值是否合法,若不合法,则抛出异常;若合法,则通过 for 循环将下标值为index 及其之后的元素向后移动一个位置,并在下标值index所在的位置添加元素,代码实现如下:
def insert(self,index,value):
'''
在顺序表中任意位置插入元素
:param index:待插入元素的位置
:param value:待插入元素的值
'''
#若index非法,则抛出异常
if index < 0 or index > self.index:
raiseIndexError('idnex非法')
#若index刚好为顺序表尾
if index = self.index:
self.append(value)
#通过for…in…将下标值为index及其之后的元素向后移动一个位置
else:
self.data += [value]
for i in range(self.index, index, -1):
self.data[i] = self.data[i-1]
self.data[index] = value
self.index += 1
删除元素前需要先判断顺序表是否为空,empty() 方法用来判断顺序表是否为空,这里可以直接调用。如果顺序表为空,则退出程序;否则将 self.index 减1,代码实现如下:
def pop(self):
'''
删除表尾元素
:return: 顺序表为空的出口
'''
if self.empty():
return
self.index += 1
先判断下标值是否合法,不合法就抛出异常;否则用for循环将index后面的所有元素都向前移动一个位置,覆盖 index 原本的值,代码实现如下:
def delete(self, index):
'''
顺序表中删除任意位置的元素
:param index:删除下标值为 index 的元素
'''
# 下标不合法,抛出异常
if self.empty() or index >= self.index:
raise IndexError('index 非法')
# 下标合法,通过for循环将index后面的所以元素都向前移动一个位置,从而覆盖index原本的值
for i in range(index, self.index):
self.data[i] = self.data[i+1]
self.index -= 1
初始化时,self.index 为0,每当向顺序表中添加一个元素时,self.index 加1;删除元素时,self.index 减 1,所以 self.index 记录着顺序表中元素的个数,代码实现如下:
def length(self):
'''
获取顺序表的长度
:return:当前顺序表中元素的个数
'''
return self.index
先判断顺序表是否为空,若顺序表为空,则抛出异常;否则遍历输入顺序表的数据元素,代码实现如下:
def traversal(self):
'''
遍历顺序表
'''
for i in range(self.index):
print(self.data[i],end=" ")
定义: 若要用一组任意的存储单元存储线性表的数据元素,则需要把这些分散的元素 ”链“ 起来,这种方法存储的线性表称为线性链表,简称链表。
特点: 存储链表中的节点的一组任意的存储单元可以是连续的,也可以是不连续的,即逻辑上相邻的数据元素在位置上不一定相邻
**实现:**为了正确表示节点间的逻辑关系,每个存储单元需要包含两部分的内容:
值域 ( d a t a ) (data) (data) : 存放节点的值
指针域 ( n e x t ) (next) (next) : 用来存放节点的直接后继指针。
这两部分组成了链表中的节点结构,即单链表存储结构,如图所示:
class Node(object):
def __init__(self, val):
# 存放节点的数据域
self.val = val
# 指向后继节点的指针
self.next = None
class SingleLinkedList(object):
def __init__(self):
'''
单链表初始化
'''
# 声明头指针、尾指针均指向空
self.head = None
self.tail = None
若表头指针指向空,则单链表为空;若表头节点不指向空,则单链表不为空,代码实现如下:
def empty(self):
'''
判断单链表是否为空
:return: 如果单链表为空返回True,否则返回False
'''
return self.head is None
遍历整个单链表,若当前节点不为空,则单链表长度加 1,代码实现如下:
def length(self):
'''
获取单链表的长度
:return:返回单链表的长度
'''
# size 用来计算单链表的长度
size = 0
# 声明 cur 指针,用来遍历单链表
cur = self.head
while cur != None:
size += 1
cur = cur.next
return size
情况一:链表为空,将头、尾指针指向新节点,新节点成为表头节点。
情况二:链表不为空,将新节点的直接后继指针指向头结点,如图所示:
代码实现如下:
def prepend(self, val):
'''
头插法
:param val: 待插入的关键字
'''
newNode = Node(val)
# 如果链表为空
if self.head is None:
# 头、尾指针指向新节点
self.head = newNode
self.tail = newNode
# 如果链表不为空
else:
# 将新节点的后继指针指向头结点
newNode.next = self.next
self.head = newNode
将新节点插入链表的第 i i i 个位置时,需要先遍历单链表到第 i − 1 i-1 i−1 个节点,将第 i − 1 i-1 i−1 个位置的节点的直接后继指针指向新节点,将新节点的直接后继指针指向第 i i i 个位置,是的新节点成为第 i i i 个位置的新节点,如图所示。
代码实现如下:
def insert(self, index, value):
'''
在链表的中间位置插入新节点
:param
index: 在下标值index 处插入元素, index 从0开始
value: 新节点的数据域
'''
# 声明指针 cur ,用来遍历链表
cur = self.head
# 遍历停止的条件, cur 指向下标为 index-1 的节点
for i in range(index-1):
cur = cur.next
# temp 指针指向原来的第 index 个节点
temp = cur.next
newNode = Node(value)
# 将新节点的后继指针指向 temp
newNode.next = temp
# 将第 index 个节点的直接后继指向新节点
cur.next = newNode
情况一:单链表为空,将头、尾指针指向新节点
情况二:单链表不为空,将表尾节点直接后继指向新节点,如图所示:
代码实现如下:
def append(self, val):
'''
尾插入法
:param val: 待插入的数据元素
'''
newNode = Node(val)
# 如果单链表为空
if self.empty():
# 头指针、尾指针指向新节点
self.head = newNode
self.tail = newNode
# 如果单链表不为空
else:
# 尾节点的后继指针指向新节点
self.tail.next = newNode
# 尾指针指向新节点
self.tail = newNode
情况一:单链表为空,无法删除,抛出异常
情况二:单链表不为空且只有一个节点,将头、尾指针指向空
情况三:单链表不为空且有多个节点,将头指针指向第二个节点,如图所示:
代码实现:
def del_first(self):
'''
删除头结点
'''
if self.empty():
raise IndexError('链表为空')
# 单链表只有一个节点
if self.length() == 1:
self.head = None
self.tail = None
# 链表长度大于1
else:
self.head = self.head.next
遍历单链表到第 i − 1 i-1 i−1 个位置的节点。将第 i − 1 i-1 i−1 个位置的节点的指针指向第 i + 1 i+1 i+1 个位置的节点。如图所示:
代码如下:(注:说下标值得时候是从0开始的,但是说“第”的时候是从1开始的,第一个第二个…元素等)
def delete(self, index):
'''
删除任意位置的节点
:param index: 要删除下标值为 index 位置的节点,
'''
if self.empty():
raise IndexError('链表为空')
# 如果 index 非法,则抛出异常
if index < 0 or index >= self.length():
raise IndexError('Index 非法')
if index == 0:
self.del_first()
else index == self.length() - 1:
self.del_last()
else:
pre = self.head
# 通过for循环使pre指针指向第index个节点
for i in range(index - 1):
pre = pre.next
pre.next = pre.next.next
情况一:单链表为空,无法删除节点,抛出异常
情况二:单链表不为空但只有一个节点,将头、尾指针指向空
情况三:单链表不为空且有多个节点,将尾指针指向导数第二个节点,并将导数第二个节点的后继指针指向空。如图所示:
代码实现:
def del_last(self):
'''
删除尾节点
'''
if self.empty():
raise IndexError('链表为空')
if self.length() == 1:
self.head = None
self.tail = None
else:
pre = self.head
cur = pre.next
while cur.next is not None:
pre = cur
cur = cur.next
cur = None
pre.next = None
self.tail = pre
在单链表中查找值为 key 的节点,从表头开始顺序查找,若存在节点的值等于key,则查找成功,否则查找失败,代码实现如下:
def find(self, key):
'''
:param key: 关键字
:return 查找成功返回True,失败返回False
'''
cur = self.head
while cur != None:
if key == cur.val:
return True
cur = cur.next
return False
空间上
时间上
**单链表的不足:**在单链表中,当前节点只能指向它的后继节点,如果要访问当前节点的前一个节点,就需要从头开始遍历。为了更方便的访问当前节点的前一个节点,引入了双链表。
**定义:**双链表是在单链表的基础上增加一个指针指向前驱节点,其存储结构如图所示。
代码实现:
class Node(object):
def __init__(self, val):
# 存放节点中的数据域
self.val = val
# 后继指针
self.next = None
# 前驱指针
self.prev = None
双链表初始化
class DoubleLinkedList(object):
def __init__(self):
'''
双链表初始化:要将链表的头指针和尾指针指向空
'''
# 声明头指针,将头指针指向空
self.head = None
# 声明尾指针,将尾指针指向空
self.tail = None
def empty(self):
'''
判断双链表是否为空
:return : 如果头指针指向空,则双链表为空,返回True,否则不为空,返回FALSE
'''
return self.head is None
遍历双链表,每经过一个节点,链表长度加一,代码实现如下:
def length(self):
'''
获取链表长度
:return: 返回链表长度
'''
# 用来记录链表长度
size = 0
# cur 指针用来遍历链表
cur = self.head
# cur 指针不指向空,表示尚未遍历到表尾节点
while cur != None:
# 链表长度加1
size += 1
# 将cur指针指向当前节点的直接后继节点
cur = cur.next
return size
头插法就是将新节点插入到表头。
如图所示:
代码实现如下:
def prepend(self, var):
'''
头插法
:param
val: 待插入的关键字
'''
# 新节点
newNode = Node(val)
# 链表为空
if self.empty():
# 将表头指针、表尾指针指向新节点
self.head = newNode
self.tail = newNode
# 链表不为空
else:
# 将头结点的前驱指针指向新节点
self.head.prev = newNode
# 将新节点的后继指针指向头结点
newNode.next = self.head
# 将表头指针指向新节点
self.head = newNode
将新节点插入到链表中的第i个位置时,首先要遍历链表到链表的第 i − 1 i-1 i−1个位置的几点,再进行两个操作:
代码实现如下:
def insert(self, index, val):
'''
:Desc
在链表任意位置添加节点,若该任意位置为index,则在第index个节点后插入元素
:param
index: 位置下标
val: 关键字
'''
# 声明pre指针
pre = self.head
newNode = Node(val)
# 将pre指针遍历到双链表中的第index个节点
for i in range(index - 1):
pre = pre.next
# 第index个节点的后继节点
next = pre.next
# 将新节点的额后继指针指向链表的第index+1 个节点
newNode.next = next
# 链表中第index + 1 个节点的寝取指针指向新节点
next。prev = newNode
# 第index个节点的后继指针指向新节点
pre.next = newNode
# 新节点的前驱指针指向链表的第index个节点
newNode.prev = pre
尾插发就是将新节点插入到表尾。
如图所示:
代码如下:
def append(self, val):
'''
尾插发
:param
val: 待插入的关键字
'''
# 新节点
newNode = Node(val)
# 如果双链表为空
if self.empty():
# 将表头指针、表尾指针指向新节点
self.head = newNode
self.tail = newNode
# 如果双链表不为空
else:
# 将尾节点的后继指针指向新节点
self.tail.next = newNode
# 将新节点的前驱指针指向尾节点
newNode.prev = self.tail
# 尾指针指向新节点,即新节点变为链表新的尾节点
self.tail = newNode
将头结点的后继指针指向空,将头结点的下一个节点的前驱指针也指向空,此时表头节点和链表分离,删除头结点操作成功。如图所示。
代码实现如下:
def del_first(self):
'''
删除头结点
'''
# 双链表为空,抛出异常
if self.empty():
raise IndexError('Index 非法')
# 双链表不为空
# 将表头指针指向头结点的后继节点,即双链表原来的第二个节点成为新节点
self.head = self.head.next
# 将新的头结点的前驱指针指向空
self.head.prev = None
在链表中遍历到第 i − 1 i-1 i−1 个节点,将第 i − 1 i-1 i−1 个节点的后继指针指向第 i + 1 i+1 i+1 个节点,将第 i i i 个节点的前驱指针指向空;将第 i i i 个节点的后继指针指向空,将第 i + 1 i+1 i+1 个节点的前驱指针指向第 i − 1 i-1 i−1 个节点,即可删除链表的第 i i i 个节点。如图所示:
代码如下:
def delet(self, index):
'''
在双链表中间任意位置删除节点
:param
index: 删除下标为index的节点,头节点下标为0
'''
pre = self.head
for i in range(index - 1):
pre = pre.next
pre.next = pre.next.next
如图所示:
代码如下:
def del_last(self):
'''
删除表尾节点
'''
if self.empty():
raise IndexError('index 非法')
# 双链表不为空
# 将表尾指针指向尾节点的前驱节点
self.tail = self.tail.prev
#将导数第二个节点的后继指针指向空,使其成为新的尾节点
self.tail.next = None
查找一个值为 k e y key key 的节点,从表头开始进行顺序查找,若节点的值等于 k e y key key , 则朝招成功;若双链表中不存在值等于 k e y key key 的节点,则朝招失败。代码实现如下:
def find(self, key):
'''
在双链表中查找关键字key
:param key: 关键字
:return : 查找成功返回True,查找失败返回False
'''
# 声明cur指针,指向头结点
cur = self.head
# 利用cur指针来遍历双链表
while cur is not None:
if key == cur.val:
return True
cur = cur.next
return False
**分类:**循环单链表、循环双链表。此处介绍循环单链表
**循环单链表的特点:**循环单链表域单链表的区别是尾节点的后继指针不是指向空,而是指向头结点,由此形成了一个环,因此,从单链表中的任何一个节点除法都能找到其他节点,这是单链表所不能实现的。
存储结构:
代码实现:
class Nde(object):
def __init__(self, val):
self.val = val
self.next = None
class CircleLinkedList(object):
def __init__(self):
'''
初始化、声明头指针、尾指针,并将它们指向空
'''
self.head = None
self.tail = None
若循环单链表的表头、表尾指针指向空,则链表为空,否则链表不为空。代码实现如下:
def empty(self):
return self.head is None
获取循环单链表,每经过一个节点,链表长度加一,获取链表长度,代码实现如下:
def length(self):
size = 0
if self.empty():
return size
cur = self.head
# cur 指向当前链表的第一个元素,即当前链表的长度从第一个元素开始计算,长度 +1
size += 1
# cur的后继节点不是头结点,即cur不指向尾节点
while cur.next is not self.head:
size += 1
cur = cur.next
return size
代码实现如下:
def append(self, val):
'''
尾插法
:param val:待插入的关键字
'''
newNode = Node(val)
if self.empty():
self.head = newNode
self.tail = newNode
else:
self.tail.next = newNode
self.tal = newNode
self.tail.next = self.head
如图所示:
代码如下:
def delete(self):
'''
删除节点
'''
if self.empty():
raise IndexError('链表为空')
if self.length() == 1:
self.head == None
self.tail = None
else:
cur = self.head
temp = cur.next
# 遍历链表,找到导数第二个节点
while temp.next is not self.head:
cur = cur.next
temp = cur.next
# 将尾指针指向导数第二个节点,使之成为新的尾节点
self.tail = cur
# 新的尾节点直接后继指向表头节点
self.tail.next = self.head
在链表中查找数值为key的节点,从头节点向后遍历,判断是否有节点的值等于key,若有,则查找成功,否则查找失败,代码实现如下:
def find(self, key):
'''
查找某个关键字
:param key: 关键字
:return : 存在则返回True,否则返回False
'''
cur = self.head
while cur.next != self.head:
if key == cur.var:
return True
cur = cur.next
return False
若链表为空,则无法遍历链表,退出程序;若链表不为空,则从表头开始向后遍历,并且打印节点的数据域,代码实现如下。
def travesal(self):
'''
遍历循环链表
'''
if self.empty():
raise IndexError('链表为空')
cur = self.head
# 当cur的直接后继指针不是指向头结点是时,即cur尚未指向尾节点时
while cur.next is not self.head:
print(cur.val, end = ' ')
cur = cur.next
# 打印尾节点
print(cur.val)