线性结构Linear Structure

❖线性结构是一种有序数据项的集合,其中 每个数据项都有唯一的前驱和后继

  除了第一个没有前驱,最后一个没有后继 新的数据项加入到数据集中时,只会加入到原有 某个数据项之前或之后 具有这种性质的数据集,就称为线性结构

 ❖线性结构总有两端,在不同的情况下,两 端的称呼也不同

  有时候称为“左”“右”端、“前”“后”端、 “顶”“底”端

 ❖两端的称呼并不是关键,不同线性结构的 关键区别在于数据项增减的方式

   有的结构只允许数据项从一端添加,而有的结构 则允许数据项从两端移除

 ❖4个最简单但功能强大的结构:栈Stack,队列Queue,双端队列Deque 和列表List

  这些数据集的共同点在于,数据项之间只存在先 后的次序关系,都是线性结构

 ❖这些线性结构是应用最广泛的数据结构 ,它们出现在各种算法中,用来解决大量重 要问题

、、、

什么是栈Stack

1.栈

❖一种有次序的数据项集合,在栈中,数据 项的加入和移除都仅发生在同一端

  这一端叫栈“顶top”,另一端叫栈“底base”

❖日常生活中有很多栈的应用

  盘子、托盘、书堆等等

❖距离栈底越近的数据项,留在栈中的时间 就越长

  而最新加入栈的数据项会被最先移除

❖这种次序通常称为“后进先出LIFO”: Last in First out

  这是一种基于数据项保存时间的次序,时间越短 的离栈顶越近,而时间越长的离栈底越近

2.栈的特性:反转次序

❖我们观察一个由混合的python原生数据 对象形成的栈

  进栈和出栈的次序正好相反

 ❖这种访问次序反转的特性,我们在某些计 算机操作上碰到过

  浏览器的“后退back”按钮,最先back的是最 近访问的网页 Word的“Undo”按钮,最先撤销的是最近操作

3.抽象数据类型Stack

❖抽象数据类型“栈”是一个有次序的数据 集,每个数据项仅从“栈顶”一端加入到 数据集中、从数据集中移除,栈具有后进 先出LIFO的特性

 ❖抽象数据类型“栈”定义为如下的操作

Stack():创建一个空栈,不包含任何数据项

push(item):将item加入栈顶,无返回值

pop():将栈顶数据项移除,并返回,栈被修改

peek():“窥视”栈顶数据项,返回栈顶的数 据项但不移除,栈不被修改

isEmpty():返回栈是否为空栈

size():返回栈中有多少个数据项

class Stack:
def __init__(self):
self.items=[] def isEmpty(self):
return self.items==[]
def push(self,item):
self.items.append(item)
def pop(self):
return self.items.pop()
def peek(self):
return self.items[len(self.items)-1]
def size(self):
return len(self.items) s=Stack() print(s.isEmpty())
s.push(4)
s.push('DOG')
print(s.peek())
s.push(True)
print(s.size())
print(s.isEmpty())
s.push(8.4)
print(s.pop())
print(s.pop())
print(s.size())

4.用Python实现ADT Stack

❖一个细节:Stack的两端对应list设置

可以将List的任意一端(index=0或者-1)设置 为栈顶 我们选用List的末端(index=-1)作为栈顶 这样栈的操作就可以通过对list的append和pop 来实现,很简单!

代码:

class Stack:
def __init__(self):
self.items=[]
def isEmpty(self):
return self.items==[]
def push(self,item):
self.items.append(item)
def pop(self):
return self.items.pop()
def peek(self):
return self.items[len(self.items)-1]
def size(self):
return len(self.items)

5.ADT Stack的另一个实现

❖不同的实现方案保持了ADT接口的稳定性

  但性能有所不同,栈顶首端的版本(左),其 push/pop的复杂度为O(n),而栈顶尾端的实现 (右),其push/pop的复杂度为O(1)

栈的应用:简单括号匹配

1.简单括号匹配

❖如何构造括号匹配识别算法

从左到右扫描括号串,最新打开的左括号,应该 匹配最先遇到的右括号 这样,第一个左括号(最早打开),就应该匹配 最后一个右括号(最后遇到) 这种次序反转的识别,正好符合栈的特性

最终代码:

class Stack:
def __init__(self):
self.items=[] def isEmpty(self):
return self.items==[]
def push(self,item):
self.items.append(item)
def pop(self):
return self.items.pop()
def peek(self):
return self.items[len(self.items)-1]
def size(self):
return len(self.items) def parChecker(symbolString):
s=Stack()
balanced=True
index=0 # 判断是否大于传进来参数的长循环次数度 与 balanced变量是否为真
# 如果达到这两个条件终止循环
while index<len(symbolString) and balanced:
symbol=symbolString[index]
#判断当前传进来的第index个字符是不是等于"C"
if symbol=="(":
#如果是push进栈
s.push(symbol)
else:
#否则判断栈是否为空
if s.isEmpty():
blanced=False
else:
#如果不空则将数据pop出去
s.pop()
index=index+1 #balanced必须为true 数组还要为空 则true
if balanced and s.isEmpty():
return True
else:
return False print(parChecker('(())'));
print(parChecker('(()'));

2.通用括号匹配

❖HTML/XML文档也有类似于括号的开闭 标记,这种层次结构化文档的校验、操作 也可以通过栈来实现

栈的应用:十进制转换为二进制

1.二进制转十进制

❖ 二进制是计算机原理中最基本的概念,作为组成 计算机最基本部件的逻辑门电路,其输入和输出 均仅为两种状态:0和1

❖ 但十进制是人类传统文化中最基本的数值概念, 如果没有进制之间的转换,人们跟计算机的交互 会相当的困难

❖所谓的“进制” ,就是用多少个字符来表 示整数

  十进制是0~9这十个数字字符,二进制是0、1两 个字符

❖十进制转换为二进制,采用的是“除以2 求余数”的算法 将整数不断除以2,每次得到的余数就是由低到 高的二进制位

❖“除以2”的过程,得到的余数是从低到 高的次序,而输出则是从高到低,所以需 要一个栈来反转次序

 十进制转二进制代码:

class Stack:
def __init__(self):
self.items=[] def isEmpty(self):
return self.items==[]
def push(self,item):
self.items.append(item)
def pop(self):
return self.items.pop()
def peek(self):
return self.items[len(self.items)-1]
def size(self):
return len(self.items) def divideBy2(decNumber):
remstack=Stack() while decNumber>0:
rem=decNumber%2
remstack.push(rem)
decNumber=decNumber // 2 binString=""
while not remstack.isEmpty():
binString=binString+str(remstack.pop())
return binString; print(divideBy2(42));

2.扩展到更多进制转换

❖十进制转换为二进制的算法,很容易可以 扩展为转换到任意N进制

  只需要将“除以2求余数”算法改为“除以N求余 数”算法就可以

❖计算机中另外两种常用的进制是八进制和 十六进制

❖主要的问题是如何表示八进制及十六进制

二进制有两个不同数字0、1

十进制有十个不同数字0、1、2、3、4、5、6、 7、8、9

八进制可用八个不同数字0、1、2、3、4、5、6 、7

十六进制的十六个不同数字则是0、1、2、3、4 、5、6、7、8、9、A、B、C、D、E、F

扩展到多进制代码:

class Stack:
def __init__(self):
self.items=[] def isEmpty(self):
return self.items==[]
def push(self,item):
self.items.append(item)
def pop(self):
return self.items.pop()
def peek(self):
return self.items[len(self.items)-1]
def size(self):
return len(self.items) def divideBy2(decNumber,base):
digits="0123456789ABCDEF"
remstack=Stack() while decNumber>0:
rem=decNumber%base
remstack.push(rem)
decNumber=decNumber // base binString=""
while not remstack.isEmpty():
binString=binString+digits[remstack.pop()]
return binString; print(divideBy2(25,2));
print(divideBy2(25,16));

什么是算法分析

11

什么是算法分析

111

学习笔记:[算法分析]数据结构与算法Python版[基本的数据结构-上]的更多相关文章

  1. 【数据结构与算法Python版学习笔记】引言

    学习来源 北京大学-数据结构与算法Python版 目标 了解计算机科学.程序设计和问题解决的基本概念 计算机科学是对问题本身.问题的解决.以及问题求解过程中得出的解决方案的研究.面对一 个特定问题,计 ...

  2. 北京大学公开课《数据结构与算法Python版》

    之前我分享过一个数据结构与算法的课程,很多小伙伴私信我问有没有Python版. 看了一些公开课后,今天特向大家推荐北京大学的这门课程:<数据结构与算法Python版>. 课程概述 很多同学 ...

  3. 学习笔记24—win10环境下python版libsvm的安装

    1.前言 由于毕业设计需要用到libsvm,所以最近专心于配置libsvm,曾经尝试过在matlab中安装,但是没有成功.最终在Python环境中完成安装. 2.LIBSVM介绍 LIBSVM 是台湾 ...

  4. 【数据结构与算法Python版学习笔记】算法分析

    什么是算法分析 算法是问题解决的通用的分步的指令的聚合 算法分析主要就是从计算资源的消耗的角度来评判和比较算法. 计算资源指标 存储空间或内存 执行时间 影响算法运行时间的其他因素 分为最好.最差和平 ...

  5. 学习笔记:[算法分析]数据结构与算法Python版

    什么是算法分析 对比程序,还是算法? ❖如何对比两个程序? 看起来不同,但解决同一个问题的程序,哪个" 更好"? ❖程序和算法的区别 算法是对问题解决的分步描述 程序则是采用某种编 ...

  6. 【数据结构与算法Python版学习笔记】查找与排序——散列、散列函数、区块链

    散列 Hasing 前言 如果数据项之间是按照大小排好序的话,就可以利用二分查找来降低算法复杂度. 现在我们进一步来构造一个新的数据结构, 能使得查找算法的复杂度降到O(1), 这种概念称为" ...

  7. 【数据结构与算法Python版学习笔记】基本数据结构——列表 List,链表实现

    无序表链表 定义 一种数据项按照相对位置存放的数据集 抽象数据类型无序列表 UnorderedList 方法 list() 创建一个新的空列表.它不需要参数,而返回一个空列表. add(item) 将 ...

  8. 【数据结构与算法Python版学习笔记】目录索引

    引言 算法分析 基本数据结构 概览 栈 stack 队列 Queue 双端队列 Deque 列表 List,链表实现 递归(Recursion) 定义及应用:分形树.谢尔宾斯基三角.汉诺塔.迷宫 优化 ...

  9. 【数据结构与算法Python版学习笔记】递归(Recursion)——定义及应用:分形树、谢尔宾斯基三角、汉诺塔、迷宫

    定义 递归是一种解决问题的方法,它把一个问题分解为越来越小的子问题,直到问题的规模小到可以被很简单直接解决. 通常为了达到分解问题的效果,递归过程中要引入一个调用自身的函数. 举例 数列求和 def ...

随机推荐

  1. GeoServer发布shapfile字段名和值乱码问题解决

    摘要: 网上说了一大堆方法又是转格式咯又是改源代码了,修改很简单: 修改Styles下的你的style: Xml代码 修改Stores下你的图层的属性,设置 DBF charset为GBK 以上设置G ...

  2. sql ROW_NUMBER() 排序函数

    1.使用row_number()函数进行编号:如 1 select email,customerID, ROW_NUMBER() over(order by psd) as rows from QT_ ...

  3. 想买保时捷的运维李先生学Java性能之 JIT即时编译器

    前言 本文记录日常学习<深入理解Java虚拟机>,不知道为啥感觉看一遍也就过了,喜欢动动手理解理解,这样才有点感觉,静不下心来的时候,看书抄书也可以用这个办法. 一.什么是JIT(Just ...

  4. Pyqy5 让窗口居中

    # QDesktopWidget import sys from PyQt5.QtWidgets import QDesktopWidget,QMainWindow,QApplication from ...

  5. Linux服务器部署redis常见问题处理

    redis开启和禁用登陆密码校验 1. 开启登陆密码校验 在redis-cli命令行工具中执行如下命令: config set requirepass yourpassword2. 禁用登陆密码校验 ...

  6. 并发编程——多线程计数的更优解:LongAdder原理分析

    前言 最近在学习ConcurrentHashMap的源码,发现它采用了一种比较独特的方式对map中的元素数量进行统计,自然是要好好研究一下其原理思想,同时也能更好地理解ConcurrentHashMa ...

  7. C语言函数的学习

    0x01.函数 这节就先讲函数吧,函数大致分为四种类型 1.无参数.无返回值的函数格式 void 函数名() { //代码段 } void Hello() { printf("Hello W ...

  8. 机器学习 第4篇:sklearn 最邻近算法概述

    sklearn.neighbors 提供了针对无监督和受监督的基于邻居的学习方法的功能.监督的基于最邻近的机器学习算法是值:对带标签的数据的分类和对连续数据的预测(回归). 无监督的最近算法是许多其他 ...

  9. debian 安裝SSH 增加新用戶 并使用sudo

    1 新建新用戶user 2 3 adduser user 4 5 passwd 123654 6 7 exit 刚安装好的Debian默认还没有sudo功能. 1.安装sudo # apt-get i ...

  10. AQS源码深入分析之共享模式-你知道为什么AQS中要有PROPAGATE这个状态吗?

    本文基于JDK-8u261源码分析 本篇文章为AQS系列文的第二篇,前文请看:[传送门] 第一篇:AQS源码深入分析之独占模式-ReentrantLock锁特性详解 1 Semaphore概览 共享模 ...