自定義語言的實現——直譯器模式(四)
18.4 完整解決方案
為了能夠解釋機器人控制指令,Sunny軟體公司開發人員使用直譯器模式來設計和實現機器人控制程式。針對五條文法規則,分別提供五個類來實現,其中終結符表示式direction、action和distance對應DirectionNode類、ActionNode類和DistanceNode類,非終結符表示式expression和composite對應SentenceNode類和AndNode類。
我們可以通過抽象語法樹來表示具體解釋過程,例如機器人控制指令“down run 10 and left move 20”對應的抽象語法樹如圖18-4所示:
圖18-4 機器人控制程式抽象語法樹例項
機器人控制程式例項基本結構如圖18-5所示:
圖18-5 機器人控制程式結構圖
在圖18-5中,AbstractNode充當抽象表示式角色,DirectionNode、ActionNode和DistanceNode充當終結符表示式角色,AndNode和SentenceNode充當非終結符表示式角色。完整程式碼如下所示:
//注:本例項對機器人控制指令的輸出結果進行模擬,將英文指令翻譯為中文指令,實際情況是呼叫不同的控制程式進行機器人的控制,包括對移動方向、方式和距離的控制等
import java.util.*;
//抽象表示式
abstract class AbstractNode {
public abstract String interpret();
}
//And解釋:非終結符表示式
class AndNode extends AbstractNode {
private AbstractNode left; //And的左表示式
private AbstractNode right; //And的右表示式
public AndNode(AbstractNode left, AbstractNode right) {
this.left = left;
this.right = right;
}
//And表示式解釋操作
public String interpret() {
return left.interpret() + "再" + right.interpret();
}
}
//簡單句子解釋:非終結符表示式
class SentenceNode extends AbstractNode {
private AbstractNode direction;
private AbstractNode action;
private AbstractNode distance;
public SentenceNode(AbstractNode direction,AbstractNode action,AbstractNode distance) {
this.direction = direction;
this.action = action;
this.distance = distance;
}
//簡單句子的解釋操作
public String interpret() {
return direction.interpret() + action.interpret() + distance.interpret();
}
}
//方向解釋:終結符表示式
class DirectionNode extends AbstractNode {
private String direction;
public DirectionNode(String direction) {
this.direction = direction;
}
//方向表示式的解釋操作
public String interpret() {
if (direction.equalsIgnoreCase("up")) {
return "向上";
}
else if (direction.equalsIgnoreCase("down")) {
return "向下";
}
else if (direction.equalsIgnoreCase("left")) {
return "向左";
}
else if (direction.equalsIgnoreCase("right")) {
return "向右";
}
else {
return "無效指令";
}
}
}
//動作解釋:終結符表示式
class ActionNode extends AbstractNode {
private String action;
public ActionNode(String action) {
this.action = action;
}
//動作(移動方式)表示式的解釋操作
public String interpret() {
if (action.equalsIgnoreCase("move")) {
return "移動";
}
else if (action.equalsIgnoreCase("run")) {
return "快速移動";
}
else {
return "無效指令";
}
}
}
//距離解釋:終結符表示式
class DistanceNode extends AbstractNode {
private String distance;
public DistanceNode(String distance) {
this.distance = distance;
}
//距離表示式的解釋操作
public String interpret() {
return this.distance;
}
}
//指令處理類:工具類
class InstructionHandler {
private String instruction;
private AbstractNode node;
public void handle(String instruction) {
AbstractNode left = null, right = null;
AbstractNode direction = null, action = null, distance = null;
Stack stack = new Stack(); //宣告一個棧物件用於儲存抽象語法樹
String[] words = instruction.split(" "); //以空格分隔指令字串
for (int i = 0; i < words.length; i++) {
//本例項採用棧的方式來處理指令,如果遇到“and”,則將其後的三個單詞作為三個終結符表示式連成一個簡單句子SentenceNode作為“and”的右表示式,而將從棧頂彈出的表示式作為“and”的左表示式,最後將新的“and”表示式壓入棧中。 if (words[i].equalsIgnoreCase("and")) {
left = (AbstractNode)stack.pop(); //彈出棧頂表示式作為左表示式
String word1= words[++i];
direction = new DirectionNode(word1);
String word2 = words[++i];
action = new ActionNode(word2);
String word3 = words[++i];
distance = new DistanceNode(word3);
right = new SentenceNode(direction,action,distance); //右表示式
stack.push(new AndNode(left,right)); //將新表示式壓入棧中
}
//如果是從頭開始進行解釋,則將前三個單片語成一個簡單句子SentenceNode並將該句子壓入棧中
else {
String word1 = words[i];
direction = new DirectionNode(word1);
String word2 = words[++i];
action = new ActionNode(word2);
String word3 = words[++i];
distance = new DistanceNode(word3);
left = new SentenceNode(direction,action,distance);
stack.push(left); //將新表示式壓入棧中
}
}
this.node = (AbstractNode)stack.pop(); //將全部表示式從棧中彈出
}
public String output() {
String result = node.interpret(); //解釋表示式
return result;
}
}
工具類InstructionHandler用於對輸入指令進行處理,將輸入指令分割為字串陣列,將第1個、第2個和第3個單片語合成一個句子,並存入棧中;如果發現有單詞“and”,則將“and”後的第1個、第2個和第3個單片語合成一個新的句子作為“and”的右表示式,並從棧中取出原先所存句子作為左表示式,然後組合成一個And節點存入棧中。依此類推,直到整個指令解析結束。
編寫如下客戶端測試程式碼:
class Client {
public static void main(String args[]) {
String instruction = "up move 5 and down run 10 and left move 5";
InstructionHandler handler = new InstructionHandler();
handler.handle(instruction);
String outString;
outString = handler.output();
System.out.println(outString);
}
}
編譯並執行程式,輸出結果如下:
向上移動5再向下快速移動10再向左移動5 |
【作者:劉偉 http://blog.csdn.net/lovelion】
相關文章
- 自定義語言的實現——直譯器模式(五)模式
- 自定義語言的實現——直譯器模式(三)模式
- 自定義語言的實現——直譯器模式(二)模式
- Java設計模式-17、直譯器模式-自定義語言的實現Java設計模式
- 實現JavaScript語言直譯器(三)JavaScript
- 源語言、目標語言、翻譯器、編譯器、直譯器編譯
- 直譯器模式模式
- 使用PHP實現詞法分析與自定義語言PHP詞法分析
- 終:直譯器模式模式
- Go 語言實現解析器翻譯Go
- 程式語言實現模式模式
- SICP第四章閱讀心得 - Lisp直譯器的實現Lisp
- 中文程式語言中文程式語言實現:翻譯器
- 24_直譯器模式模式
- Python直譯器簡介(4):動態語言Python
- 設計模式之直譯器模式設計模式
- 設計模式(十五):直譯器模式設計模式
- Rust語言之GoF設計模式: 直譯器Interpreter模式RustGo設計模式
- 魔獸爭霸自定義地圖直譯器存在安全漏洞地圖
- 設計模式學習筆記(二十二)直譯器模式及其實現設計模式筆記
- 從編譯原理看一個直譯器的實現編譯原理
- Django高階程式設計之自定義Field實現多語言Django程式設計
- 深入 WebAssembly 之直譯器實現篇Web
- 直譯器模式(Interpreter)模式
- 設計模式(十五)直譯器設計模式
- 簡說設計模式——直譯器模式設計模式
- 極簡設計模式-直譯器模式設計模式
- Python設計模式-直譯器模式Python設計模式
- JAVA設計模式之直譯器模式Java設計模式
- SAPGUI裡實現自定義的語法檢查GUI
- 設計模式--直譯器模式和狀態模式設計模式
- 什麼是程式語言,什麼是Python直譯器Python
- 化繁為簡的翻譯機——直譯器模式模式
- 自定義web伺服器(四)Web伺服器
- 用 Python 實現 Python 直譯器Python
- Golang實現微型數學運算直譯器Golang
- 直譯器構造實現函式呼叫函式
- 23種設計模式之直譯器模式設計模式