解釋器模式 (Interpreter Pattern):給定一個語言,定義它的文法的一種表示,並定義一個解釋器,這個解釋器使用該表示來解釋語言中的句子。若是一種特定類型的問題發生的頻率足夠高,那麼可能就值得將該問題的各個實例表述爲一個簡單語言中的句子。這樣就能夠構建一個解釋器,該解釋器經過解釋這些句子來解決該問題。ios
再拿上面的遊戲這個例子進行說明,我能夠定義如下五條文法:express
expression ::= direction action distance | composite //表達式 composite ::= expression 'and' expression //複合表達式 direction ::= 'up' | 'down' | 'left' | 'right' //移動方向 action ::= 'move' | 'walk' //移動方式 distance ::= an integer //移動距離
AbstractExpression:聲明一個抽象的解釋操做,這個接口被抽象語法樹中全部的節點所共享;工具
TernimalExpression:一個句子中的每一個終結符須要該類的一個實例,它實現與文法中的終結符相關聯的解釋操做;spa
NonternimalExpression:翻譯
Context:包含解釋器以外的一些全局信息;
Client:構建一個須要進行解釋操做的文法句子,而後調用解釋操做進行解釋。調試
實際進行解釋時,按照如下時序進行的:code
#include <string> #include <iostream> using namespace std; /************************************************************************ * description: 演奏內容 * remark: ************************************************************************/ class playContext { public: string getPlayText() { return m_strText; } void setPlayText(const string& strText) { m_strText = strText; } private: string m_strText; }; /************************************************************************ * description: 表達式類 * remark: ************************************************************************/ 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 = stoi(strPlayValue); nPos = PlayContext.getPlayText().find(" "); //遞歸處理 PlayContext.setPlayText(PlayContext.getPlayText().substr(nPos + 1)); //cout << PlayContext.getPlayText() << endl; excute(strPlayKey, nPlayValue); } } // 執行 virtual void excute(string& strKey, const int nValue) = 0; }; /************************************************************************ * description: 音符類 * remark: ************************************************************************/ class note : public expression { public: virtual void excute(string& strKey, const int nValue) { char szKey = strKey.substr(0, 1)[0]; string strNote; switch (szKey) { case 'C': strNote = "N1"; break; case 'D': strNote = "N2"; break; case 'E': strNote = "N3"; break; case 'F': strNote = "N4"; break; case 'G': strNote = "N5"; break; case 'A': strNote = "N6"; break; case 'B': strNote = "N7"; break; default: strNote = "error"; break; } cout << strNote << " "; } }; /************************************************************************ * description: 音階類 * remark: ************************************************************************/ 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: }; //音樂速度類 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 << " "; } }; int main() { playContext context; cout << "Music:"; context.setPlayText("T 2 O 2 E 2 G 3 G 4 "); expression* expressObj = nullptr; while (!context.getPlayText().empty()) { char szKey = context.getPlayText().substr(0, 1)[0]; switch (szKey) { 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 (nullptr != expressObj) { expressObj->interpret(context); } } system("pause"); return 0; }