Python 数据结构预算法之栈(Stack)的实现与应用

栈的数据结构模型:

栈是一种有次序的数据项集合, 在栈中, 数据项的加入和移除都仅发生在同一端,这一端叫栈顶,另一端叫栈底。

距离栈底越近的数据项, 留在栈中的时间就越长,而最新加入栈的数据项会被最先移除,也即“后进先出LIFO”。

使用Python实现ADT Stack:

选用最常用的数据集list来实现,选用list的末端(index=-1)作为栈顶,就可以通过对list的append和pop来实现栈的push和pop,会比较简单。

class Stack:
    def __init__(self):
        self.items = []

    def is_empty(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)


栈的应用:

  • 简单括号匹配
import Stack

'''
简单括号匹配
输入字符串只包含 ( ) [ ] { }
输出True False
'''


def parChecker(symbolString):
    s = Stack.Stack()
    balanced = True
    index = 0
    while index < len(symbolString) and balanced:
        symbol = symbolString[index]
        if symbol in "([{":
            s.push(symbol)
        else:
            if s.is_empty():
                balanced = False
            else:
                top = s.pop()
                if not match(top, symbol):
                    balanced = False
        index = index + 1
    if balanced and s.is_empty():
        return True
    else:
        return False


def match(left, right):
    opens = "([{"
    closes = ")]}"
    return opens.index(left) == closes.index(right)


print(parChecker('([])'))
print(parChecker('()[]{}'))
print(parChecker('(([{]}))'))
  • 进制转换
from Stack import Stack
'''
进制转换
输入10进制数字以及需要转换到的进制位数
输出转换后的对应进制的字符串
'''

def baseConvert(decNumber, base):
    digit = "0123456789ABCDEF"
    tranStack = Stack()
    while decNumber > 0:
        rem = decNumber % base
        tranStack.push(rem)
        decNumber = decNumber // base
    newString= ""
    while not tranStack.is_empty():
        newString = newString + digit[tranStack.pop()]
    return newString


print(baseConvert(13, 8))
print(baseConvert(13, 16))
  • 表达式转换(中缀转后缀)
    思路:
  1. 将中缀表达式转换为单词(token)列表
  2. 从左到右逐个字符扫描中缀表达式的过程中,采用一个opstack栈来暂存未处理的操作符。
    • 如果单词是操作数,则直接添加到后缀表达式列表的末尾。
    • 如果单词是"(",则压入opstack栈顶。
    • 如果单词是")",则反复弹出opstack栈顶操作符,加入到输出列表末尾,直到碰到左括号。
    • 如果单词是操作符"*/+-",则压入opstack的栈顶。但在压入之前,要比较其与栈顶操作符的优先级,如果栈顶的高于或者等于它,就要反复弹出栈顶操作符,加入到输出列表末尾。
  3. 中缀表达式单词列表扫描结束后,把opstack栈中的所有剩余操作符依次弹出,添加到输出列表末尾。
  4. 把输出列表再用join方法合并成后缀表达式字符串,算法结束。
from Stack import Stack

def infixToPostfix(infixExp):

    opStack = Stack()
    expList = []
    indexToken = {}
    indexToken["("] = 0
    indexToken[")"] = 0
    indexToken["+"]= 1
    indexToken["-"] = 1
    indexToken["*"] = 2
    indexToken["/"] = 2
    for token in infixExp:
        if token in "ABCDEFGHIJKLMNOPQRSTUVWXYZ" or token in "123456789":
            expList.append(token)
        elif token is "(":
            opStack.push(token)
        elif token is ")":
            topToken = opStack.pop()
            while topToken is not "(":
                expList.append(topToken)
                topToken = opStack.pop()
        elif token in "+-*/":
            while (not opStack.is_empty()) and (indexToken[opStack.peek()] >= indexToken[token]):
                    expList.append(opStack.pop())
            opStack.push(token)

    while not opStack.is_empty():
        expList.append(opStack.pop())

    return "".join(expList)

print(infixToPostfix("3+2*5"))
print(infixToPostfix("(3+2)*5"))
print(infixToPostfix("(3+2*5)*5"))
  • 后缀表达式求值
    思路:
  1. 创建空栈opStack暂存操作数。
  2. 从左到右逐个字符扫描后缀表达式的过程中,
    • 如果单词是操作数,则先转换为int,然后压入opStack中。
    • 如果单词是"+-*/",则取出opstack栈顶的两个元素进行操作,先弹出右操作数后弹出左操作数,计算后的值压入栈中。
  3. 单词列表扫描完成,表达式的值在栈顶。
from Stack import Stack


def postFixCal(postFixStr):

    opStack = Stack()
    for token in postFixStr:
        if token in "123456789":
            opStack.push(int(token))
        else:
            rightOps = opStack.pop()
            leftOps = opStack.pop()
            result = doMath(token,int(rightOps),int(leftOps))
            opStack.push(result)

    return opStack.pop()


def doMath(token,rightOps,leftOps):

    if token is "+":
        return rightOps+leftOps
    elif token is "-":
        return rightOps - leftOps
    elif token is "*":
        return rightOps * leftOps
    elif token is "/":
        return rightOps / leftOps

print(postFixCal("325*+"))
print(postFixCal("325*+5*"))
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 217,084评论 6 503
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,623评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 163,450评论 0 353
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,322评论 1 293
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,370评论 6 390
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,274评论 1 300
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,126评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,980评论 0 275
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,414评论 1 313
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,599评论 3 334
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,773评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,470评论 5 344
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,080评论 3 327
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,713评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,852评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,865评论 2 370
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,689评论 2 354