數(shù)據(jù)結(jié)構(gòu)入門(mén)(二)棧的應(yīng)用之?dāng)?shù)學(xué)表達(dá)式求值

??在文章數(shù)據(jù)結(jié)構(gòu)入門(mén)(一)棧的實(shí)現(xiàn)中稠茂,我們已經(jīng)知道了如何去實(shí)現(xiàn)“椖”這個(gè)數(shù)據(jù)結(jié)構(gòu)桩引,并且介紹了一個(gè)它的應(yīng)用:數(shù)的進(jìn)制轉(zhuǎn)換。在本文中澎语,將會(huì)介紹棧的第二個(gè)應(yīng)用,也就是棧在數(shù)學(xué)表達(dá)式求值中的應(yīng)用。
??我們分以下幾步對(duì)數(shù)學(xué)表達(dá)式進(jìn)行求值擅羞。

  • 棧的實(shí)現(xiàn)尸变;
  • 中綴表達(dá)式轉(zhuǎn)后綴表達(dá)式;
  • 后綴表達(dá)式求值减俏。

先不著急明白上述術(shù)語(yǔ)召烂,你看下去就會(huì)明白了。

棧的實(shí)現(xiàn)

??以下是棧的Python實(shí)現(xiàn)(Stack.py)娃承,代碼如下:

# -*- coding: utf-8 -*-

class Empty(Exception):
    # Error attempting to access an element from an empty container
    pass

class Stack:
    # initialize
    def __init__(self):
        self.__data = []

    # length of Stack
    def __len__(self):
        return len(self.__data)

    # whether the Stack is empty
    def is_empty(self):
        return len(self.__data) == 0

    # push an element is Stack
    def push(self, e):
        self.__data.append(e)

    # top element of Stack
    def top(self):
        if self.is_empty():
            raise Empty('Stack is empty')
        return self.__data[-1]

    # remove the top element of Stack
    def pop(self):
        if self.is_empty():
            raise Empty('Stack is empty')
        return self.__data.pop()

    # clear the Stack
    def clear(self):
        while not self.is_empty():
            self.pop()

中綴表達(dá)式轉(zhuǎn)后綴表達(dá)式

??首先奏夫,我們來(lái)看一下數(shù)學(xué)表達(dá)式的三種形式:前綴表達(dá)式,中綴表達(dá)式历筝,后綴表達(dá)式酗昼。
??中綴表達(dá)式(Infix Expression)就是我們平時(shí)常用的書(shū)寫(xiě)方式,帶有括號(hào)漫谷。前綴表達(dá)式(Prefix Expression)要求運(yùn)算符出現(xiàn)在運(yùn)算數(shù)字的前面仔雷,后綴表達(dá)式(Postfix Expression)要求運(yùn)算符出現(xiàn)在運(yùn)算數(shù)字的后面,一般這兩種表達(dá)式不出現(xiàn)括號(hào)舔示。示例如下:

中綴表達(dá)式 前綴表達(dá)式 后綴表達(dá)式
A + B * C + D + + A * B C D A B C * + D +
(A + B) * (C + D) * + A B + C D A B + C D + *
A * B + C * D + * A B * C D A B * C D * +
A + B + C + D + + + A B C D A B + C + D +

一般在計(jì)算機(jī)中碟婆,為了方便對(duì)表達(dá)式求值,我們需要將熟悉的中綴表達(dá)式轉(zhuǎn)化為后綴表達(dá)式惕稻。
??中綴表達(dá)式轉(zhuǎn)后綴表達(dá)式的算法如下:

  1. 創(chuàng)建一個(gè)空棧opstack竖共,用于儲(chǔ)存運(yùn)算符。創(chuàng)建一個(gè)空的列表俺祠,用于儲(chǔ)存輸出結(jié)果公给。
  2. 將輸入的中綴表達(dá)式(字符串形式)用字符串的split方法轉(zhuǎn)化為一個(gè)列表。
  3. 從左到右對(duì)該列表進(jìn)行遍歷操作(元素為token)蜘渣,如下:
    • 如果token為運(yùn)算數(shù)淌铐,則將它添加(append)至輸出列表中。
    • 如果token為左小括號(hào)蔫缸,則將它壓入(psuh)到opstack中腿准。
    • 如果token是右小括號(hào),則對(duì)opstack進(jìn)行pop操作拾碌,直至對(duì)應(yīng)的左小括號(hào)被移出吐葱。將移出的運(yùn)算符添加(append)到輸出列表的末端。
    • 如果token是 *, /, +, -, 中的一個(gè)校翔,則將其壓入(push)到opstack中弟跑。注意,先要移除那些運(yùn)算優(yōu)先級(jí)大于等于該token的運(yùn)算符防症,并將它們添加到輸出列表中孟辑。
  4. 當(dāng)上述過(guò)程結(jié)果后哎甲,檢查opstack。任何還在opstack中的運(yùn)算符都應(yīng)移除扑浸,并將移出的運(yùn)算符添加(append)到輸出列表的末端烧给。

??上述過(guò)程的完整Python代碼如下:

# -*- coding: utf-8 -*-
from Stack import Stack

# 中綴表達(dá)式轉(zhuǎn)化為后綴表達(dá)式
def infixToPostfix(infixexpr):
    prec = {"*": 3, "/": 3, "+": 2, "-": 2, "(": 1}
    opStack = Stack()
    postfixList = []
    tokenList = infixexpr.split()

    for token in tokenList:
        if token.isdigit() or '.' in token:
            postfixList.append(token)
        elif token == '(':
            opStack.push(token)
        elif token == ')':
            topToken = opStack.pop()
            while topToken != '(':
                postfixList.append(topToken)
                topToken = opStack.pop()
        else:
            while (not opStack.is_empty()) and (prec[opStack.top()] >= prec[token]):
                  postfixList.append(opStack.pop())
            opStack.push(token)

    while not opStack.is_empty():
        postfixList.append(opStack.pop())
    return " ".join(postfixList)

# inExpr = "( ( 1 + 2 ) * 3 ) * ( 3 - 1.2 )"
inExpr = "10 + 3 * 5 / ( 16 - 4 )"
postExpr = infixToPostfix(inExpr)
print(postExpr)

輸出結(jié)果如下:

10 3 5 * 16 4 - / +

后綴表達(dá)式求值

??當(dāng)把中綴表達(dá)式轉(zhuǎn)化為后綴表達(dá)式之后,我們?cè)倮脳?duì)后綴表達(dá)式求值喝噪。其具體的算法如下:

  1. 建立一個(gè)棧來(lái)存儲(chǔ)待計(jì)算的運(yùn)算數(shù)础嫡;
  2. 遍歷字符串,遇到運(yùn)算數(shù)則壓入棧中酝惧,遇到運(yùn)算符則出棧運(yùn)算數(shù)(2次)榴鼎,進(jìn)行相應(yīng)的計(jì)算,計(jì)算結(jié)果是新的操作數(shù)晚唇,壓入棧中巫财,等待計(jì)算;
  3. 按上述過(guò)程哩陕,遍歷完整個(gè)表達(dá)式平项,棧中只剩下最終結(jié)果;

??完整的Python代碼如下:(接以上代碼)

# -*- coding: utf-8 -*-
from Stack import Stack

# 兩個(gè)數(shù)的運(yùn)算, 除法時(shí)分母不為0
def operate(op, num1, num2):
    if num2 == 0:
        raise ZeroDivisionError
    else:
        res = {
                '+': num1 + num2,
                '-': num1 - num2,
                '*': num1 * num2,
                '/': num1 / num2,
              }
        return res[op]

# 將字符串轉(zhuǎn)化為浮點(diǎn)型或整型數(shù)字
def str2num(s):
    if '.' in s:
        return float(s)
    else:
        return int(s)

# 后綴表達(dá)式求值
def evalPostfix(e):

    tokens = e.split()  # 后綴表達(dá)式轉(zhuǎn)化為列表
    s = Stack()
    for token in tokens:
        if token.isdigit() or '.' in token:  # 如果當(dāng)前元素是數(shù)字
            s.push(str2num(token))
        elif token in '+-*/':  # 如果當(dāng)前元素是運(yùn)算符
            op2 = s.pop()
            op1 = s.pop()
            s.push(operate(token, op1, op2))  # 計(jì)算結(jié)果入棧
    return s.pop()

# inExpr = "( ( 1 + 2 ) * 3 ) * ( 3 - 1.2 )"
inExpr = "10 + 3 * 5 / ( 16 - 4 )"
postExpr = infixToPostfix(inExpr)
print(postExpr)
result = evalPostfix(postExpr)
print(result)

輸出結(jié)果:

11.25

請(qǐng)務(wù)必注意悍及,我們輸入的中綴表達(dá)式中闽瓢,每個(gè)運(yùn)算符或運(yùn)算符要用空格隔開(kāi)。

參考文獻(xiàn)

  1. 3.9. Infix, Prefix and Postfix Expressions: http://interactivepython.org/runestone/static/pythonds/BasicDS/InfixPrefixandPostfixExpressions.html
  2. Python算法實(shí)戰(zhàn)系列之棧: http://python.jobbole.com/87581/

注意:本人現(xiàn)已開(kāi)通微信公眾號(hào): Python爬蟲(chóng)與算法(微信號(hào)為:easy_web_scrape)心赶, 歡迎大家關(guān)注哦~~

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末扣讼,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子缨叫,更是在濱河造成了極大的恐慌椭符,老刑警劉巖,帶你破解...
    沈念sama閱讀 218,122評(píng)論 6 505
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件耻姥,死亡現(xiàn)場(chǎng)離奇詭異销钝,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī)琐簇,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,070評(píng)論 3 395
  • 文/潘曉璐 我一進(jìn)店門(mén)曙搬,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人鸽嫂,你說(shuō)我怎么就攤上這事≌鹘玻” “怎么了据某?”我有些...
    開(kāi)封第一講書(shū)人閱讀 164,491評(píng)論 0 354
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)诗箍。 經(jīng)常有香客問(wèn)我癣籽,道長(zhǎng),這世上最難降的妖魔是什么? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 58,636評(píng)論 1 293
  • 正文 為了忘掉前任筷狼,我火速辦了婚禮瓶籽,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘埂材。我一直安慰自己塑顺,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,676評(píng)論 6 392
  • 文/花漫 我一把揭開(kāi)白布俏险。 她就那樣靜靜地躺著严拒,像睡著了一般。 火紅的嫁衣襯著肌膚如雪竖独。 梳的紋絲不亂的頭發(fā)上裤唠,一...
    開(kāi)封第一講書(shū)人閱讀 51,541評(píng)論 1 305
  • 那天,我揣著相機(jī)與錄音莹痢,去河邊找鬼种蘸。 笑死,一個(gè)胖子當(dāng)著我的面吹牛竞膳,可吹牛的內(nèi)容都是我干的航瞭。 我是一名探鬼主播,決...
    沈念sama閱讀 40,292評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼顶猜,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼沧奴!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起长窄,我...
    開(kāi)封第一講書(shū)人閱讀 39,211評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤滔吠,失蹤者是張志新(化名)和其女友劉穎,沒(méi)想到半個(gè)月后挠日,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體疮绷,經(jīng)...
    沈念sama閱讀 45,655評(píng)論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,846評(píng)論 3 336
  • 正文 我和宋清朗相戀三年嚣潜,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了冬骚。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 39,965評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡懂算,死狀恐怖只冻,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情计技,我是刑警寧澤喜德,帶...
    沈念sama閱讀 35,684評(píng)論 5 347
  • 正文 年R本政府宣布,位于F島的核電站垮媒,受9級(jí)特大地震影響舍悯,放射性物質(zhì)發(fā)生泄漏航棱。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,295評(píng)論 3 329
  • 文/蒙蒙 一萌衬、第九天 我趴在偏房一處隱蔽的房頂上張望饮醇。 院中可真熱鬧,春花似錦秕豫、人聲如沸朴艰。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 31,894評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)呵晚。三九已至,卻和暖如春沫屡,著一層夾襖步出監(jiān)牢的瞬間饵隙,已是汗流浹背。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 33,012評(píng)論 1 269
  • 我被黑心中介騙來(lái)泰國(guó)打工沮脖, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留金矛,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 48,126評(píng)論 3 370
  • 正文 我出身青樓勺届,卻偏偏與公主長(zhǎng)得像驶俊,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子免姿,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,914評(píng)論 2 355

推薦閱讀更多精彩內(nèi)容