深入解析C++設(shè)計(jì)模式編程中解釋器模式的運(yùn)用
解釋器模式(interpreter),給定一個(gè)語(yǔ)言,定義它的文法的一種表示,并定義一個(gè)解釋器,這個(gè)解釋器使用該表示來(lái)解釋語(yǔ)言中的句子。
解釋器模式需要解決的是,如果一種特定類(lèi)型的問(wèn)題發(fā)生的頻率足夠高,那么可能就值得將該問(wèn)題的各個(gè)實(shí)例表述為一個(gè)簡(jiǎn)單語(yǔ)言中的句子。這樣就可以構(gòu)建一個(gè)解釋器,該解釋器通過(guò)解釋這些句子來(lái)解決該問(wèn)題。當(dāng)有一個(gè)語(yǔ)言需要解釋執(zhí)行,并且你可將該語(yǔ)言中的句子表示為一個(gè)抽象語(yǔ)法樹(shù)時(shí),可使用解釋器模式。用了解釋器模式,就意味著可以很容易地改變和擴(kuò)展文法,因?yàn)樵撃J绞褂妙?lèi)來(lái)表示文法規(guī)則,你可使用繼承來(lái)改變或擴(kuò)展該文法。也比較容易實(shí)現(xiàn)文法,因?yàn)槎x抽象語(yǔ)法樹(shù)中各個(gè)節(jié)點(diǎn)的類(lèi)的實(shí)現(xiàn)大體類(lèi)似,這些類(lèi)都易于直接編寫(xiě)。
結(jié)構(gòu)圖:
實(shí)例:
音樂(lè)解釋器
playContext.h
/************************************************************************ * description: 演奏內(nèi)容 * remark: ************************************************************************/ #ifndef _PLAY_CONTEXT_H_ #define _PLAY_CONTEXT_H_ #include <string> #include <iostream> using namespace std; class playContext { public: string getPlayText() { return m_strText; } void setPlayText(const string& strText) { m_strText = strText; } private: string m_strText; }; #endif// _PLAY_CONTEXT_H_
expression.h
/************************************************************************ * description: 表達(dá)式類(lèi) * remark: ************************************************************************/ #ifndef _EXPRESSION_H_ #define _EXPRESSION_H_ #include "playContext.h" class expression { public: // 解釋器 void interpret(playContext& PlayContext) { if (PlayContext.getPlayText().empty()) { return; } else { string strPlayKey = PlayContext.getPlayText().substr(0, 1); string strtemp = PlayContext.getPlayText().substr(2); PlayContext.setPlayText(strtemp); size_t nPos = PlayContext.getPlayText().find(" "); string strPlayValue = PlayContext.getPlayText().substr(0, nPos); int nPlayValue = atoi(strPlayValue.c_str()); nPos = PlayContext.getPlayText().find(" "); PlayContext.setPlayText(PlayContext.getPlayText().substr(nPos + 1)); excute(strPlayKey, nPlayValue); } } // 執(zhí)行 virtual void excute(string& strKey, const int nValue) = 0; private: }; #endif// _EXPRESSION_H_
note.h
/************************************************************************ * description: 音符類(lèi) * remark: ************************************************************************/ #ifndef _NOTE_H_ #define _NOTE_H_ #include "expression.h" class note : public expression { public: virtual void excute(string& strKey, const int nValue) { char szKey[2]; strncpy(szKey, strKey.c_str(), strKey.length()); string strNote; switch (szKey[0]) { case 'C': strNote = "1"; break; case 'D': strNote = "2"; break; case 'E': strNote = "3"; break; case 'F': strNote = "4"; break; case 'G': strNote = "5"; break; case 'A': strNote = "6"; break; case 'B': strNote = "7"; break; default: strNote = "error"; break; } cout << strNote << " "; } }; #endif// _NOTE_H_
scale.h
/************************************************************************ * description: 音階類(lèi) * remark: ************************************************************************/ #ifndef _SCALE_H_ #define _SCALE_H_ #include "expression.h" class scale : public expression { public: virtual void excute(string& strKey, const int nValue) { string strScale; switch (nValue) { case 1: strScale = "低音"; break; case 2: strScale = "中音"; break; case 3: strScale = "高音"; break; default: strScale = "error"; break; } cout << strScale << " "; } private: }; #endif// _SCALE_H_
speed.h
#ifndef _SPEED_H_ #define _SPEED_H_ #include "expression.h" class speed : public expression { public: virtual void excute(string& strKey, const int nValue) { string strSpeed; if (nValue < 3) { strSpeed = "快速"; } else if (nValue >= 6) { strSpeed = "慢速"; } else { strSpeed = "中速"; } cout << strSpeed << " "; } }; #endif// _SPEED_H_
客戶(hù)端: InterpreterApp.cpp
// InterpreterApp.cpp : 定義控制臺(tái)應(yīng)用程序的入口點(diǎn)。 // #include "stdafx.h" #include "note.h" #include "scale.h" #include "speed.h" #include "playContext.h" int _tmain(int argc, _TCHAR* argv[]) { playContext context; cout << "Music:"; context.setPlayText("T 2 O 2 E 3 G 5 G 5 "); expression* expressObj = NULL; while (!context.getPlayText().empty()) { string strSep = context.getPlayText().substr(0, 1); char szKey[2]; strncpy(szKey, strSep.c_str(), strSep.length()); switch (szKey[0]) { case 'O': expressObj = new scale(); break; case 'T': expressObj = new speed(); break; case 'C': case 'D': case 'E': case 'F': case 'G': case 'A': case 'B': case 'P': expressObj = new note(); break; default: break; } if (NULL != expressObj) { expressObj->interpret(context); } } system("pause"); return 0; }
不足之處
解釋器模式不足的是,解釋器模式為文法中的每一條規(guī)則至少定義了一個(gè)類(lèi),因此包含許多規(guī)則的文法可能難以管理和維護(hù)。建議當(dāng)文法非常復(fù)雜時(shí),使用其他的技術(shù)如語(yǔ)法分析程序或編譯器生成器來(lái)處理。
適用場(chǎng)景
- 當(dāng)有一個(gè)語(yǔ)言需要解釋執(zhí)行, 并且你可將該語(yǔ)言中的句子表示為一個(gè)抽象語(yǔ)法樹(shù)時(shí),可使用解釋器模式。而當(dāng)存在以下情況時(shí)該模式效果最好:
- 該文法簡(jiǎn)單對(duì)于復(fù)雜的文法, 文法的類(lèi)層次變得龐大而無(wú)法管理。此時(shí)語(yǔ)法分析程序生成器這樣的工具是更好的選擇。它們無(wú)需構(gòu)建抽象語(yǔ)法樹(shù)即可解釋表達(dá)式, 這樣可以節(jié)省空間而且還可能節(jié)省時(shí)間。
- 效率不是一個(gè)關(guān)鍵問(wèn)題最高效的解釋器通常不是通過(guò)直接解釋語(yǔ)法分析樹(shù)實(shí)現(xiàn)的, 而是首先將它們轉(zhuǎn)換成另一種形式。例如,正則表達(dá)式通常被轉(zhuǎn)換成狀態(tài)機(jī)。但即使在這種情況下, 轉(zhuǎn)換器仍可用解釋器模式實(shí)現(xiàn), 該模式仍是有用的。
- 實(shí)例講解C++設(shè)計(jì)模式編程中State狀態(tài)模式的運(yùn)用場(chǎng)景
- 解析C++編程中如何使用設(shè)計(jì)模式中的狀態(tài)模式結(jié)構(gòu)
- 詳解C++設(shè)計(jì)模式編程中對(duì)訪問(wèn)者模式的運(yùn)用
- C++設(shè)計(jì)模式編程之Flyweight享元模式結(jié)構(gòu)詳解
- 詳解設(shè)計(jì)模式中的中介者模式在C++編程中的運(yùn)用
- 詳解C++設(shè)計(jì)模式編程中責(zé)任鏈模式的應(yīng)用
- 詳解設(shè)計(jì)模式中的Command命令模式及相關(guān)C++實(shí)現(xiàn)
- C++設(shè)計(jì)模式編程中使用Bridge橋接模式的完全攻略
- 設(shè)計(jì)模式中的備忘錄模式解析及相關(guān)C++實(shí)例應(yīng)用
- C++設(shè)計(jì)模式編程中簡(jiǎn)單工廠與工廠方法模式的實(shí)例對(duì)比
相關(guān)文章
C++11智能指針unique_ptr用法使用場(chǎng)景分析
unique_ptr 是 C++ 11 提供的用于防止內(nèi)存泄漏的智能指針中的一種實(shí)現(xiàn),即使在異常發(fā)生時(shí)也可幫助避免資源泄露。這篇文章主要介紹了C++11智能指針unique_ptr用法介紹,需要的朋友可以參考下2021-08-08編寫(xiě)C++程序使DirectShow進(jìn)行視頻捕捉
這篇文章主要介紹了如何編寫(xiě)C++程序來(lái)使DirectShow進(jìn)行視頻捕捉的方法,DirectShow是微軟公司在ActiveMovie和Video for Windows的基礎(chǔ)上推出的新一代基于COM(Component Object Model)的流媒體處理的開(kāi)發(fā)包,要的朋友可以參考下2016-03-03C++實(shí)現(xiàn)編寫(xiě)二維碼的示例代碼
這篇文章主要為大家詳細(xì)介紹如何基于C++實(shí)現(xiàn)編寫(xiě)二維碼的功能,文中的示例代碼講解詳細(xì),具有一定的學(xué)習(xí)價(jià)值,感興趣的小伙伴可以了解一下2023-06-06詳解C語(yǔ)言動(dòng)態(tài)內(nèi)存的分配
這篇文章主要為大家介紹了C語(yǔ)言動(dòng)態(tài)內(nèi)存的分配,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下,希望能夠給你帶來(lái)幫助2021-12-12利用OpenCV實(shí)現(xiàn)局部動(dòng)態(tài)閾值分割
這篇文章主要為大家詳細(xì)介紹了利用OpenCV實(shí)現(xiàn)局部動(dòng)態(tài)閾值分割,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-01-01淺談C++ Explicit Constructors(顯式構(gòu)造函數(shù))
下面小編就為大家?guī)?lái)一篇淺談C++ Explicit Constructors(顯式構(gòu)造函數(shù))。小編覺(jué)得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2016-12-12