數(shù)據(jù)結(jié)構(gòu)--棧的應(yīng)用--括號(hào)匹配問(wèn)題和行編輯器

括號(hào)匹配校驗(yàn)

假設(shè)表達(dá)式中允許包含兩種括號(hào)又碌,圓括號(hào)和方括號(hào),其嵌套順序隨意裕循,即[()[]]椎扬、[([][])][]()[]等為正確格式,[(])([())等均為不正確格式谓晌。要求編寫(xiě)一個(gè)程序檢驗(yàn)括號(hào)輸入是否正確别垮。

思路整理

此題我們使用棧的后進(jìn)先出的原則來(lái)實(shí)現(xiàn),思路如下:

  1. 如果以])開(kāi)頭那么括號(hào)肯定是不匹配的扎谎。
  2. 將接收到的字符串拆分成單個(gè)字符
  3. 遍歷入棧
    • 若當(dāng)前椞枷耄空,則直接入棧
    • 若當(dāng)前棧不為空毁靶,則取出棧頂數(shù)據(jù)與當(dāng)前數(shù)據(jù)做對(duì)比胧奔。若兩個(gè)括號(hào)匹配,則取出棧頂數(shù)據(jù)预吆。否則將當(dāng)前數(shù)據(jù)入棧
  4. 遍歷完成后檢查棧是否為空龙填。若為空則說(shuō)明校驗(yàn)成功,否則檢驗(yàn)失敗


    956be32c0c934679801e1600a530d56cpng

代碼實(shí)現(xiàn)


package com.codestd.study.stack;

import java.util.Stack;
/**
 * 括號(hào)校驗(yàn)
 * @author jaune
 * @since 1.0.0
 */
public class BracketChecker {

    /**
     * 校驗(yàn)括號(hào)是否正確
     * @param in 輸入括號(hào)字符串
     */
    public static boolean check(String in) {
        char[] chars = in.toCharArray();
        Stack<Character> stack = new Stack<>();
        for (char aChar : chars) {
            // 如果不是括號(hào)字符直接丟棄
            if (isBracket(aChar)) {
                if (stack.isEmpty()) {
                    stack.push(aChar);
                } else {
                    char stackChar = stack.peek();
                    // 注意這里的順序,一定是棧中的是前括號(hào)岩遗,比較的是后括號(hào)才行扇商。順序不能亂。)( 這種格式是錯(cuò)誤的宿礁。
                    if (checkPairBracket(stackChar, aChar)) {
                        stack.pop();
                    } else {
                        stack.push(aChar);
                    }
                }
            }
        }

        return stack.isEmpty();
    }

    /**
     * 檢驗(yàn)前括號(hào)和后括號(hào)是否匹配
     * @param ch1 前括號(hào)
     * @param ch2 后括號(hào)
     */
    public static boolean checkPairBracket(char ch1, char ch2) {
        return ch1 == '(' && ch2 == ')' || ch1 == '[' && ch2 == ']';
    }

    /**
     * 檢查是否為括號(hào)字符
     */
    public static boolean isBracket(char ch) {
        return ch == '(' || ch == ')' || ch == '[' || ch == ']';
    }
}

此題中只說(shuō)了小括號(hào)和中括號(hào)案铺,如果再加上大括號(hào),或者書(shū)名號(hào)等成對(duì)出現(xiàn)的字符梆靖,核心的代碼不用做修改控汉,只需要修改checkPairBracketisBracket即可。無(wú)論多少種括號(hào)返吻,校驗(yàn)的原理都是一樣的姑子。

測(cè)試代碼如下:

import static org.assertj.core.api.Assertions.*;

/**
 * Test for {@link BracketChecker}
 */
public class BracketCheckerTest {

    @Test
    public void test() {
        String s1 = "[]()[(())()]";
        assertThat(BracketChecker.check(s1)).isTrue();
        String s2 = ")[]()[]";
        assertThat(BracketChecker.check(s2)).isFalse();
        String s3 = "[(])[]";
        assertThat(BracketChecker.check(s3)).isFalse();
    }
}

以上測(cè)試代碼已測(cè)試通過(guò)。由于棧的底層實(shí)現(xiàn)是數(shù)組或鏈表测僵,所以使用數(shù)組的方式也可以解決此問(wèn)題街佑。只需要一個(gè)指針指向數(shù)組尾部數(shù)據(jù)即可。

行編輯程序

一個(gè)簡(jiǎn)單的行編輯器程序的功能是:接收用戶從終端輸入的程序或數(shù)據(jù)捍靠,對(duì)用戶輸入的一行數(shù)據(jù)做處理舆乔,#表示退格鍵,刪除#前的一個(gè)字符剂公。@表示@符號(hào)之前的數(shù)據(jù)均無(wú)效希俩。

比如用戶輸入了 gosd##od實(shí)際有效字符是good,當(dāng)用戶輸入了pgm@progre#am實(shí)際有效的是program纲辽。

思路分析

將一行的字符依次讀入棧中颜武,遇到#取出棧頂數(shù)據(jù),若遇到@則清空棧拖吼。

實(shí)現(xiàn)


import java.util.Arrays;
import java.util.Stack;
import java.util.stream.Collectors;

/**
 * 行編輯器
 *
 * @author jaune
 */
public class LineEditor {

    public String handle(String line) {
        char[] chars = line.toCharArray();
        Stack<Character> stack = new Stack<>();
        for (char ch : chars) {
            if (stack.isEmpty()) {
                if (ch != '#' && ch != '@') {
                    stack.push(ch);
                }
            } else {
                if (ch == '#') {
                    stack.pop();
                } else if (ch == '@') {
                    stack.clear();
                } else {
                    stack.push(ch);
                }
            }
        }
        Character[] res = new Character[stack.size()];
        stack.toArray(res);
        stack.clear();
        return Arrays.stream(res).map(String::valueOf).collect(Collectors.joining());
    }
}
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末鳞上,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子吊档,更是在濱河造成了極大的恐慌篙议,老刑警劉巖,帶你破解...
    沈念sama閱讀 212,718評(píng)論 6 492
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件怠硼,死亡現(xiàn)場(chǎng)離奇詭異鬼贱,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī)香璃,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,683評(píng)論 3 385
  • 文/潘曉璐 我一進(jìn)店門(mén)这难,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人葡秒,你說(shuō)我怎么就攤上這事姻乓∏兑纾” “怎么了?”我有些...
    開(kāi)封第一講書(shū)人閱讀 158,207評(píng)論 0 348
  • 文/不壞的土叔 我叫張陵蹋岩,是天一觀的道長(zhǎng)赖草。 經(jīng)常有香客問(wèn)我,道長(zhǎng)剪个,這世上最難降的妖魔是什么秧骑? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 56,755評(píng)論 1 284
  • 正文 為了忘掉前任,我火速辦了婚禮禁偎,結(jié)果婚禮上腿堤,老公的妹妹穿的比我還像新娘阀坏。我一直安慰自己如暖,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,862評(píng)論 6 386
  • 文/花漫 我一把揭開(kāi)白布忌堂。 她就那樣靜靜地躺著盒至,像睡著了一般。 火紅的嫁衣襯著肌膚如雪士修。 梳的紋絲不亂的頭發(fā)上枷遂,一...
    開(kāi)封第一講書(shū)人閱讀 50,050評(píng)論 1 291
  • 那天,我揣著相機(jī)與錄音棋嘲,去河邊找鬼酒唉。 笑死,一個(gè)胖子當(dāng)著我的面吹牛沸移,可吹牛的內(nèi)容都是我干的痪伦。 我是一名探鬼主播,決...
    沈念sama閱讀 39,136評(píng)論 3 410
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼雹锣,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼网沾!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起蕊爵,我...
    開(kāi)封第一講書(shū)人閱讀 37,882評(píng)論 0 268
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤辉哥,失蹤者是張志新(化名)和其女友劉穎,沒(méi)想到半個(gè)月后攒射,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體醋旦,經(jīng)...
    沈念sama閱讀 44,330評(píng)論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,651評(píng)論 2 327
  • 正文 我和宋清朗相戀三年会放,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了浑度。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 38,789評(píng)論 1 341
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡鸦概,死狀恐怖箩张,靈堂內(nèi)的尸體忽然破棺而出甩骏,到底是詐尸還是另有隱情,我是刑警寧澤先慷,帶...
    沈念sama閱讀 34,477評(píng)論 4 333
  • 正文 年R本政府宣布饮笛,位于F島的核電站,受9級(jí)特大地震影響论熙,放射性物質(zhì)發(fā)生泄漏福青。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 40,135評(píng)論 3 317
  • 文/蒙蒙 一脓诡、第九天 我趴在偏房一處隱蔽的房頂上張望无午。 院中可真熱鬧,春花似錦祝谚、人聲如沸宪迟。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 30,864評(píng)論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)次泽。三九已至,卻和暖如春席爽,著一層夾襖步出監(jiān)牢的瞬間意荤,已是汗流浹背。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 32,099評(píng)論 1 267
  • 我被黑心中介騙來(lái)泰國(guó)打工只锻, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留玖像,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 46,598評(píng)論 2 362
  • 正文 我出身青樓齐饮,卻偏偏與公主長(zhǎng)得像捐寥,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子沈矿,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,697評(píng)論 2 351

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