成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專(zhuān)欄INFORMATION COLUMN

如何寫(xiě)一個(gè)計(jì)算器?

zsy888 / 3396人閱讀

摘要:考慮這樣一個(gè)問(wèn)題,給定一個(gè)字符串,,如何將它轉(zhuǎn)化為如下形式換句話說(shuō),就是如何將字符串按照四則運(yùn)算計(jì)算出來(lái),如何寫(xiě)一個(gè)計(jì)算器。語(yǔ)義分析使用語(yǔ)法樹(shù)檢查源程序是否和語(yǔ)言定義的語(yǔ)義一致。

考慮這樣一個(gè)問(wèn)題,給定一個(gè)字符串,“1+1+(3+4)-2*3+8/2”,如何將它轉(zhuǎn)化為如下形式:

“1+1=2”

“3+4=7”

“2+7=9”

“2*3=6”

“9-6=3”

“8/2=4”

“3+4=7”

換句話說(shuō),就是如何將字符串按照四則運(yùn)算計(jì)算出來(lái),如何寫(xiě)一個(gè)計(jì)算器。

拿 java 來(lái)舉例,并且為了簡(jiǎn)單,我們只考慮個(gè)位數(shù)。這個(gè)過(guò)程大概分為這幾個(gè)步驟,首先需要掃描字符串去除空白字符,其次將各個(gè)字符轉(zhuǎn)換成對(duì)應(yīng)的操作符或操作數(shù),然后按照四則運(yùn)算規(guī)則逐次計(jì)算并輸出。

好,我們首先構(gòu)造一個(gè) scanner,它主要功能是順序掃描字符串,返回字符并跳過(guò)其中的空白字符,如下

2015年就要結(jié)束了

public class Scanner {

    public Scanner(String source){
       this.source = source.toCharArray();
    }

    private char[] source;
    private int index = 0;
    private static char END = "
";
    public char getNext(){
        char result;

        do{
            if (index >= source.length){
                return END;
            }
            result = source[index];
            index += 1;
        }while (Character.isWhitespace(result));

        return result;
    }

}

在進(jìn)行下一步之前,讓我們思考一下這個(gè)算式的規(guī)律,算式中存在兩種對(duì)象,一種是數(shù)字,一種是操作符,由于存在運(yùn)算的優(yōu)先級(jí),我們分成三種對(duì)象,并用下面的形式來(lái)說(shuō)明。

expr —> term + expr | term - expr | term
term —> factor * term | factor/term | factor
factor—> digit |(expr)

—>的意思是由...組成, 代表或關(guān)系,expr 代表加減法運(yùn)算式,term 代表乘除法運(yùn)算式,factor 代表操作的最小元素,最后一句的意思就是 factor 由數(shù)字或者帶括號(hào)的 expr 組成。這三個(gè)定義式是遞歸的,它可以代表任意深度的算式。讓我們用樹(shù)的形式來(lái)觀察一下:

有了這三種抽象對(duì)象我們可以寫(xiě)出對(duì)應(yīng)方法了,我們?cè)趐arser類(lèi)里定義三個(gè)函數(shù),來(lái)代表三種對(duì)象的產(chǎn)生過(guò)程,并且定義char類(lèi)型變量head代表正在被掃描的字符。

public class Parser {
    private Scanner scanner;
    public Parser(Scanner scanner){
        this.scanner = scanner;
    }

    private char head;

    public void parse(){
        if (Scanner.END != (head = scanner.getNext())){
            expr();
        }
        if (head != Scanner.END){
            throw new RuntimeException(“syntax error at "+head);
        }
    }

    public int expr(){
        int result = term();
        int tempResult;
        char operate;
        while ((operate = head) == "+" || operate == "-") {
            head = scanner.getNext();
            tempResult = term();
            switch (operate) {
                case "+":
                    System.out.println(result + "+" + tempResult + "=" + (result + tempResult));
                    result += tempResult;
                    break;
                case "-":
                    System.out.println(result + "-" + tempResult + "=" + (result - tempResult));
                    result -= tempResult;
            }
        }
        return result;
    }
    public int term(){
        int result = factor();
        int tempResult;
        char operate ;
        while ((operate=head) == "*" ||operate == "/") {
            head = scanner.getNext();
            tempResult = factor();
            switch (operate) {
                case "*":
                    System.out.println(result + "*" + tempResult + "=" + (result * tempResult));
                    result *= tempResult;
                    break;
                case "/":
                    System.out.println(result + "/" + tempResult + "=" + (result / tempResult));
                    result /= tempResult;
            }
        }
        return result;
    }

    public int factor(){
        int factor;

        if (Character.isDigit(head)){
            factor = head - 48; //字符變量-48可以轉(zhuǎn)換成對(duì)應(yīng)的字面數(shù)字
            head = scanner.getNext();
        } else {
            match("(");
            factor = expr();
            match(")");

        }
        return factor;
    }

//match 方法用來(lái)斷言 head 是什么字符,如果為真,將讀取下一個(gè)字符賦值給 head

public boolean match(char symbol){
    if (symbol == head){
        head = scanner.getNext();
        return true;
    }
    throw new  RuntimeException("syntax error at "+head);
}

public static void main(String... args){
    Scanner scanner = new Scanner("1+1+(3+4)-2*3+8/2");
    Parser parser = new Parser(scanner);
    parser.parse();
}

}

如果回過(guò)頭來(lái)重新考慮這件事情,你會(huì)發(fā)現(xiàn)我們這個(gè)小程序的本質(zhì)是將一段文本轉(zhuǎn)化成可以執(zhí)行的程序,正如我們的編譯器一樣。而實(shí)際上編譯器要復(fù)雜的多,它的基本工作過(guò)程可以分為幾個(gè)步驟:

詞法分析 (scanning),讀入源程序字符流,將字符轉(zhuǎn)換成有意義的詞素 (lexeme) 的序列,并生成對(duì)應(yīng)的詞法單元 (token)

語(yǔ)法分析 (parsing),主要目的是生成詞法單元的語(yǔ)法結(jié)構(gòu),一般會(huì)使用樹(shù)形結(jié)構(gòu)來(lái)表示,稱(chēng)為語(yǔ)法樹(shù)。

語(yǔ)義分析 (semantic analysis),使用語(yǔ)法樹(shù)檢查源程序是否和語(yǔ)言定義的語(yǔ)義一致。其中一個(gè)重要部分是類(lèi)型檢查。

生成中間代碼,語(yǔ)義分析完成后,編譯器會(huì)將語(yǔ)法樹(shù)生成為一種接近機(jī)器語(yǔ)言的中間代碼。我們程序最后產(chǎn)生的一系列小的表達(dá)式與之類(lèi)似。

代碼優(yōu)化,編譯器會(huì)嘗試改進(jìn)中間代碼,用以生成更高效的機(jī)器代碼。

代碼生成,將優(yōu)化過(guò)對(duì)中間代碼生成機(jī)器代碼。

在這些過(guò)程中,遞歸的方法起到了非常重要的作用,有一句話說(shuō)明了編譯器的本質(zhì),編譯器就是讓你的源程序變成可執(zhí)行程序的另一個(gè)程序。你會(huì)發(fā)現(xiàn)這個(gè)定義本身就是遞歸的。透過(guò)這些編譯原理,可以讓我們更加深入的理解編程語(yǔ)言,甚至發(fā)明一種編程語(yǔ)言。

OneAPM Mobile Insight 以真實(shí)用戶(hù)體驗(yàn)為度量標(biāo)準(zhǔn)進(jìn)行 Crash 分析,監(jiān)控網(wǎng)絡(luò)請(qǐng)求及網(wǎng)絡(luò)錯(cuò)誤,提升用戶(hù)留存。訪問(wèn) OneAPM 官方網(wǎng)站感受更多應(yīng)用性能優(yōu)化體驗(yàn),想閱讀更多技術(shù)文章,請(qǐng)?jiān)L問(wèn) OneAPM 官方技術(shù)博客。
本文轉(zhuǎn)自 OneAPM 官方博客

文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請(qǐng)注明本文地址:http://systransis.cn/yun/65347.html

相關(guān)文章

  • 用PHP寫(xiě)一個(gè)最簡(jiǎn)單的解釋器Part5(計(jì)算器最后一節(jié),下節(jié)開(kāi)始如何寫(xiě)個(gè)腳本語(yǔ)言)

    摘要:經(jīng)過(guò)幾天的努力,用已經(jīng)實(shí)現(xiàn)了一個(gè)完整的基礎(chǔ)計(jì)算器,如下圖上代碼定義整數(shù)類(lèi)型描述定義操作符號(hào)類(lèi)型描述加法定義操作符號(hào)類(lèi)型描述減法定義操作符號(hào)類(lèi)型描述乘法定義操作符號(hào)類(lèi)型描述除法定義操作符號(hào)類(lèi)型描述定義操作符號(hào)類(lèi)型描述定義空格用來(lái)存儲(chǔ)輸入字符的 showImg(https://segmentfault.com/img/bVbdNO5?w=900&h=377); 經(jīng)過(guò)幾天的努力,用PHP已經(jīng)...

    yanest 評(píng)論0 收藏0
  • 如何寫(xiě)懶惰的代碼

    摘要:考慮下面這段的代碼現(xiàn)在的問(wèn)題是是否計(jì)算答案大概是是的。如果你需要代碼變得懶一些,你需要繞過(guò)它的勤奮。這樣的代碼太差了??上У氖牵m然是懶惰的,它卻很勤奮的計(jì)算它的參數(shù)。里有一億個(gè)數(shù),所以這樣的代碼太貴了。 懶和勤奮:懶是一個(gè)很廣泛的詞,在程序員的世界里,它指只做需要做的工作。而勤奮在這里指做很多的工作為了未來(lái)。 考慮下面這段 JavaScript 的代碼: showImg(https:...

    betacat 評(píng)論0 收藏0
  • 如何一個(gè)移動(dòng)端的聯(lián)動(dòng)選擇器(四)

    摘要:寫(xiě)在前面之前寫(xiě)了一篇為移動(dòng)端而生的自定義多級(jí)聯(lián)動(dòng)選擇器,得到了很多人的關(guān)注。具體實(shí)現(xiàn)步驟如下先傳入一個(gè)需要計(jì)算深度的對(duì)象給,判斷如果還有則迭代,并計(jì)算深度。如果增加了聯(lián)動(dòng)級(jí)數(shù)需要來(lái)判斷,則為新增加的聯(lián)動(dòng)綁定新的事件。 寫(xiě)在前面 之前寫(xiě)了一篇 MultiPicker -『為移動(dòng)端而生』的自定義多級(jí)聯(lián)動(dòng)選擇器,得到了很多人的關(guān)注。鑒于很多人對(duì)這種手寫(xiě)插件的過(guò)程很好奇,所以寫(xiě)了幾篇文章,來(lái)說(shuō)...

    ssshooter 評(píng)論0 收藏0
  • 如何一個(gè)移動(dòng)端的聯(lián)動(dòng)選擇器(四)

    摘要:寫(xiě)在前面之前寫(xiě)了一篇為移動(dòng)端而生的自定義多級(jí)聯(lián)動(dòng)選擇器,得到了很多人的關(guān)注。具體實(shí)現(xiàn)步驟如下先傳入一個(gè)需要計(jì)算深度的對(duì)象給,判斷如果還有則迭代,并計(jì)算深度。如果增加了聯(lián)動(dòng)級(jí)數(shù)需要來(lái)判斷,則為新增加的聯(lián)動(dòng)綁定新的事件。 寫(xiě)在前面 之前寫(xiě)了一篇 MultiPicker -『為移動(dòng)端而生』的自定義多級(jí)聯(lián)動(dòng)選擇器,得到了很多人的關(guān)注。鑒于很多人對(duì)這種手寫(xiě)插件的過(guò)程很好奇,所以寫(xiě)了幾篇文章,來(lái)說(shuō)...

    cgspine 評(píng)論0 收藏0
  • 如何一個(gè)移動(dòng)端的聯(lián)動(dòng)選擇器(四)

    摘要:寫(xiě)在前面之前寫(xiě)了一篇為移動(dòng)端而生的自定義多級(jí)聯(lián)動(dòng)選擇器,得到了很多人的關(guān)注。具體實(shí)現(xiàn)步驟如下先傳入一個(gè)需要計(jì)算深度的對(duì)象給,判斷如果還有則迭代,并計(jì)算深度。如果增加了聯(lián)動(dòng)級(jí)數(shù)需要來(lái)判斷,則為新增加的聯(lián)動(dòng)綁定新的事件。 寫(xiě)在前面 之前寫(xiě)了一篇 MultiPicker -『為移動(dòng)端而生』的自定義多級(jí)聯(lián)動(dòng)選擇器,得到了很多人的關(guān)注。鑒于很多人對(duì)這種手寫(xiě)插件的過(guò)程很好奇,所以寫(xiě)了幾篇文章,來(lái)說(shuō)...

    leiyi 評(píng)論0 收藏0

發(fā)表評(píng)論

0條評(píng)論

最新活動(dòng)
閱讀需要支付1元查看
<