数组、链表、堆栈、队列、二叉树——数据结构

数据结构:

   是指相互之间存在一种或多种特定关系的数据元素的集合。听起来是不是很抽象,简单理解:数据结构就是描述对象间逻辑关系的学科。比如:队列就是一种先进先出的逻辑结构,栈是一种先进后出的逻辑结构,家谱是一种树形的逻辑结构!

数据存储结构:

   它是计算机的一个概念,简单讲,就是描述数据在计算机中存储方式的学科;常用的数据存储方式就两种:顺序存储,非顺序存储!顺序存储就是把数据存储在一块连续的存储介质(比如硬盘或内存)上—-举个例子:从内存中拿出第100个字节到1000个字节间的连续位置,存储数据;数组就是典型的顺序存储!非顺序存储就是各个数据不一定存在一个连续的位置上,只要每个数据知道它前面的数据和后面的数据,就能把所有数据连续起来啦;链表就是典型的非顺序存储啦!
   数组、链表、堆栈和队列是最基本的数据结构,任何程序都会涉及到其中的一种或多种。
   数据结构: (数组、链表、堆栈、队列也在里面)

   下面简单描述一下数组、链表、堆栈、队列和二叉树

1 数组

   数组是最最基本的数据结构,很多语言都内置支持数组。数组是使用一块连续的内存空间保存数据,保存的数据的个数在分配内存的时候就是确定的:

这里写图片描述

图 1.1 包含 n 个数据的数组

   访问数组中第 n 个数据的时间花费是 O(1) 但是要在数组中查找一个指定的数据则是 O(N)。当向数组中插入或者删除数据的时候,最好的情况是在数组的末尾进行操作,时间复杂度是O(1) ,但是最坏情况是插入或者删除第一个数据,时间复杂度是 O(N) 。在数组的任意位置插入或者删除数据的时候,后面的数据全部需要移动,移动的数据还是和数据个数有关所以总体的时间复杂度仍然是 O(N) 。

数组、链表、堆栈、队列、二叉树——数据结构_第1张图片

图 1.2 向数组中插入数据

2 链表

  链表是在非连续的内存单元中保存数据,并且通过指针将各个内存单元链接在一起,最有一个节点的指针指向 NULL 。链表不需要提前分配固定大小存储空间,当需要存储数据的时候分配一块内存并将这块内存插入链表中。
  在链表中查找第 n 个数据以及查找指定的数据的时间复杂度是 O(N) ,但是插入和删除数据的时间复杂度是 O(1) ,因为只需要调整指针就可以:
这里写图片描述

图 2.1 链表

数组、链表、堆栈、队列、二叉树——数据结构_第2张图片

图 2.2 向链表中插入一个数据

数组、链表、堆栈、队列、二叉树——数据结构_第3张图片

图 2.3 从链表中删除一个数据

    向上面这样的链表结构在插入和删除的时候编程会比较困难,因为需要记住当前节点的前一个节点,这样才能完成插入和删除。为了简便通常使用带有头节点的链表:
数组、链表、堆栈、队列、二叉树——数据结构_第4张图片

图 2.4 带有头节点的单链表

   上面的链表是单链表,此外还有双链表,就是节点中包含指向下一个节点的指针和指向上一个节点的指针:
数组、链表、堆栈、队列、二叉树——数据结构_第5张图片

图 2.5 双向链表

   不带有头节点的双向链表在插入和删除数据的时候也不会出现单链表那样的问题。此外还有一种链表是循环链表,它是将双向链表的头尾相接:
数组、链表、堆栈、队列、二叉树——数据结构_第6张图片

图 2.6 双向循环链表

   向循环双向链表和循环链表中插入或者从中删除数据只是多移动几个指针。

3 堆栈

   堆栈实现了一种后进先出的语义 (LIFO) 。可以使用数组或者是链表来实现它:
数组、链表、堆栈、队列、二叉树——数据结构_第7张图片

图 3.1 堆栈

   对于堆栈中的数据的所有操作都是在栈的顶部完成的,只可以查看栈顶部的数据,只能够向栈的顶部压入数据,也只能从栈的顶部弹出数据。

4 队列

   队列实现了先入先出的语义 (FIFO) 。队列也可以使用数组和链表来实现:

这里写图片描述

图 4.1 队列

   队列只允许在队尾添加数据,在队头删除数据。但是可以查看队头和队尾的数据。还有一种是双端队列,在两端都可以插入和删除:

数组、链表、堆栈、队列、二叉树——数据结构_第8张图片

图 4.2 双端队列

5 树

数组、链表、堆栈、队列、二叉树——数据结构_第9张图片

二叉树

二叉树的简单实现:

class Node:
    def __init__(self,item):
        self.item = item
        self.child1 = None
        self.child2 = None
class Tree:
    def __init__(self):
        self.root = None

    def add(self, item):
        node = Node(item)
        if self.root is None:
            self.root = node
        else:
            q = [self.root]

            print("y", self.root)
            while True:
                print("x",q)
                print([x.item for x in q])
                pop_node = q.pop(0)
                print(pop_node.item)
                if pop_node.child1 is None:
                    pop_node.child1 = node
                    return
                elif pop_node.child2 is None:
                    pop_node.child2 = node
                    return
                else:
                    q.append(pop_node.child1)
                    q.append(pop_node.child2)

    def traverse(self):  # 层次遍历
        if self.root is None:
            return None
        q = [self.root]
        res = [self.root.item]
        while q != []:
            pop_node = q.pop(0)
            if pop_node.child1 is not None:
                q.append(pop_node.child1)
                res.append(pop_node.child1.item)

            if pop_node.child2 is not None:
                q.append(pop_node.child2)
                res.append(pop_node.child2.item)
        return res

    def preorder(self,root):  # 先序遍历
        if root is None:
            return []
        result = [root.item]
        left_item = self.preorder(root.child1)
        right_item = self.preorder(root.child2)
        return result + left_item + right_item

    def inorder(self,root):  # 中序序遍历
        if root is None:
            return []
        result = [root.item]
        left_item = self.inorder(root.child1)
        right_item = self.inorder(root.child2)
        return left_item + result + right_item

    def postorder(self,root):  # 后序遍历
        if root is None:
            return []
        result = [root.item]
        left_item = self.postorder(root.child1)
        right_item = self.postorder(root.child2)
        return left_item + right_item + result

t = Tree()
for i in range(10):
    t.add(i)
print('层序遍历:',t.traverse())
print('先序遍历:',t.preorder(t.root))
print('中序遍历:',t.inorder(t.root))
print('后序遍历:',t.postorder(t.root))
#输出结果
层次遍历: [0, 1, 2, 3, 4, 5, 6, 7, 8, 9]
先次遍历: [0, 1, 3, 7, 8, 4, 9, 2, 5, 6]
中次遍历: [7, 3, 8, 1, 9, 4, 0, 5, 2, 6]
后次遍历: [7, 8, 3, 9, 4, 1, 5, 6, 2, 0]

你可能感兴趣的:(数据结构与算法)