設計模式之筆記--解釋器模式(Interpreter)

来源:http://www.cnblogs.com/zhou-yi/archive/2016/05/06/5462663.html
-Advertisement-
Play Games

解釋器模式(Interpreter) 定義 解釋器模式(Interpreter),給定一個語言,定義它的文法的一種表示,並定義一個解釋器,這個解釋器使用該表示來解釋語言中的句子。 類圖 描述 Expression:抽象表達式,聲明一個所有的具體表達式都需要實現的抽象介面;這個介面主要是一個inter ...


解釋器模式(Interpreter)

定義

      解釋器模式(Interpreter),給定一個語言,定義它的文法的一種表示,並定義一個解釋器,這個解釋器使用該表示來解釋語言中的句子。

類圖

描述

Expression:抽象表達式,聲明一個所有的具體表達式都需要實現的抽象介面;這個介面主要是一個interpret()方法,稱做解釋操作。

Terminal Expression:終結符表達式,實現了抽象表達式所要求的介面;文法中的每一個終結符都有一個具體終結表達式與之相對應。比如公式R=R1+R2,R1和R2就是終結符,對應的解析R1和R2的解釋器就是終結符表達式。

Nonterminal Expression:非終結符表達式,文法中的每一條規則都需要一個具體的非終結符表達式,非終結符表達式一般是文法中的運算符或者其他關鍵字,比如公式R=R1+R2中,“+"就是非終結符,解析“+”的解釋器就是一個非終結符表達式。

Context:環境,它的任務一般是用來存放文法中各個終結符所對應的具體值,比如R=R1+R2,給R1賦值100,給R2賦值200,這些信息需要存放到環境中。

應用場景

首先輸入一個加減或乘除的運算公式,比如a+b-c+a或a*b/c*a,再給每個參數賦值,最後根據公式完成運算並得到結果。

 

    /// <summary>
    /// 環境
    /// </summary>
    public class Context
    {
        private Dictionary<char, double> variable;
        public Dictionary<char, double> Variable 
        {
            get
            {
                if (this.variable == null)
                {
                    this.variable = new Dictionary<char, double>();
                }
                return this.variable;
            }
        }
    }

    /// <summary>
    /// 抽象表達式
    /// </summary>
    public abstract class Expression
    {
        public abstract double Interpret(Context context);
    }

    /// <summary>
    /// 變數,終結符表達式
    /// </summary>
    public class VariableExpression : Expression
    {
        private char key;
        public VariableExpression(char key)
        {
            this.key = key;
        }

        public override double Interpret(Context context)
        {
            return context.Variable[this.key];
        }
    }

    /// <summary>
    /// 操作符,非終結符表達式
    /// </summary>
    public abstract class OperatorExpression : Expression
    {
        protected Expression left;
        protected Expression right;

        public OperatorExpression(Expression left, Expression right)
        {
            this.left = left;
            this.right = right;
        }
    }

    public class AddExpression : OperatorExpression
    {
        public AddExpression(Expression left, Expression right)
            : base(left, right)
        { 
        
        }

        public override double Interpret(Context context)
        {
            return this.left.Interpret(context) + this.right.Interpret(context);
        }
    }

    public class SubExpression : OperatorExpression
    {
        public SubExpression(Expression left, Expression right)
            : base(left, right)
        {

        }

        public override double Interpret(Context context)
        {
            return this.left.Interpret(context) - this.right.Interpret(context);
        }
    }

    public class MulExpression: OperatorExpression
    {
        public MulExpression(Expression left, Expression right)
            : base(left, right)
        { 
        
        }

        public override double Interpret(Context context)
        {
            return this.left.Interpret(context) * this.right.Interpret(context);
        }
    }

    public class DivExpression: OperatorExpression
    {
        public DivExpression(Expression left, Expression right)
            : base(left, right)
        { 
        
        }

        public override double Interpret(Context context)
        {
            return this.left.Interpret(context) / this.right.Interpret(context);
        }
    }

    public class Calculator
    {
        private string expression;
        private Context context;

        public Calculator(string expression)
        {
            this.expression = expression;
            this.context = new Context();
        }

        public double Calculate()
        {
            char[] vars = this.expression.ToCharArray();
            foreach (char c in vars)
            {
                if (c == '+' || c == '-' || c == '*' || c == '/')
                {
                    continue;
                }
                if (!this.context.Variable.ContainsKey(c))
                {
                    Console.Write(c + "=");
                    this.context.Variable.Add(c, double.Parse(Console.ReadLine()));
                }
            }
            Expression left = new VariableExpression(vars[0]);
            Expression right = null;
            Stack<Expression> stack = new Stack<Expression>();
            stack.Push(left);
            for (int i = 1; i < vars.Length; i += 2)
            {
                left = stack.Pop();
                right = new VariableExpression(vars[i + 1]);
                switch (vars[i])
                {
                    case '+':
                        stack.Push(new AddExpression(left, right));
                        break;
                    case '-':
                        stack.Push(new SubExpression(left, right));
                        break;
                    case '*':
                        stack.Push(new MulExpression(left, right));
                        break;
                    case '/':
                        stack.Push(new DivExpression(left, right));
                        break;
                }
            }
            double value = stack.Pop().Interpret(this.context);
            stack.Clear();
            return value;
        }
    }

輸入公式:a+b-c+a

賦值:

a=10
b=5
c=3

運算結果:22


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 問題: 有N件物品和一個容量為V的背包。第i件物品的價值是c[i],重量是w[i]。求解將哪些物品裝入背包可使這些物品的重量總和不超過背包容量,且價值總和最大。 這個問題的特點是:每種物品只有一件,可以選擇放或者不放。用f[i][j]表示背包當前容量為j,選擇裝入1-i個物品時的最大價值 在求最優解 ...
  • discuz x3在DIY模塊中調用偽靜態不成功,顯示動態鏈接,然而其他的鏈接正常顯示偽靜態。 後臺啟用偽靜態後,發現論壇版塊、帖子點擊鏈接,偽靜態正常顯示,然後在門戶首頁DIY顯示的帖子,點進去後發現鏈接還是動態的鏈接,不宜被搜索引擎抓取。 解決辦法: 第一步:前臺管理員登錄 --> DIY -- ...
  • HibernateCallback是回調函數,是一個介面,該介面只有一個方法doInHibernate(org.hibernate.Session session), 這個方法體就是Spring執行的持久化操作。(上面的函數,是用hirbernate的回調函數來刪除id,因為類型是int,所以最後r ...
  • 原文:http://www.imsiren.com/archives/535 為了以後能開發PHP擴展..就一定要瞭解PHP的執行順序..這篇文章就是為C開發PHP擴展做鋪墊. web環境 我們假設為 apache. 在編譯PHP的時候,為了能夠讓Apache支持PHP,我們會生成一個mod_php ...
  • python支持對mysql的操作 已經安裝配置成功python、mysql 之後根據各自電腦配置選擇對應系統的MySQL-python 文件是EXE格式、打開下一步即可 下載地址博主分享下: http://pan.baidu.com/s/1c2uhVwc 安裝成功、在IDLE中輸入: import ...
  • [Architecture] 系統架構正交分解法 前言 隨著企業成長,支持企業業務的軟體,也會越來越龐大與複雜。當系統複雜到一定程度,開發人員會發現很多系統架構的設計細節,很難有條理、有組織的用一張大藍圖去做分析設計。先前在InfoQ上看到一篇文章:「億級用戶下的新浪微博平臺架構 - 衛向軍」,在這 ...
  • 關鍵的設計原則 在開始設計之前,思考一下關鍵的原則,將會幫助你創建一個最小花費、高可用性和擴展性的架構。 分離關註點,將應用劃分為在功能上儘可能不重覆的功能點。主要的參考因素就是最小化交互,高內聚、低耦合。但是,錯誤的分離功能邊界,可能會導致功能之間的高耦合性和複雜性, 職責單一,每一個組件或者是模 ...
  • 簡單工廠模式是屬於創建型模式,又叫做靜態工廠方法(Static Factory Method)模式,但不屬於23種GOF設計模式之一。簡單工廠模式是由一個工廠對象決定創建出哪一種產品類的實例。簡單工廠模式是工廠模式家族中最簡單實用的模式,可以理解為是不同工廠模式的一個特殊實現。 簡單工廠模式的UML ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...