主頁 > .NET開發 > 編碼技巧 --- 如何實作字串運算運算式的計算

編碼技巧 --- 如何實作字串運算運算式的計算

2023-07-12 08:55:12 .NET開發

引言

最近做一個配置的功能,需求是該配置項跟另一個整形配置項關聯,具有一定的函式關系,例如有一個配置項是值為 N ,則另一配置 F 項滿足函式關系\(F=2/(N+1)\),這個函式關系是客戶手動輸入,只需要簡單的四則運算,所以我們要做的就是判斷四則運算運算式是否有效,且給定 N 的值,算出運算式的值,

如何快速判斷一個四則運算公式字串是否符合規則,且根據給定值計算出該公式的值?

雙堆疊實作

實際上編譯器就是利用了雙堆疊實作了的運算式求值,其中一個堆疊用來保存運算元,另一個堆疊用來保存運算子,

從左向右遍歷運算式,當遇到數字時,就將其直接壓入運算元堆疊;當遇到運算子時,就將其與運算子堆疊的堆疊頂元素比較,

如果遇到的運算子比運算子堆疊頂的元素的優先級高,就將這個運算子壓入堆疊;

如果遇到的運算子比運算子堆疊頂的元素的優先級低或兩者相同,就從運算子堆疊頂取出運算子,在從運算元堆疊頂取兩個運算元,然后進行計算,并把計算的得到的結果壓入運算元堆疊,繼續比較這個運算子與運算子堆疊頂的元素;

下圖表示一個簡單四則運算運算式 3+5*8-6的計算程序:

image.png

代碼實作可以大概簡化可以分為以下步驟:

  1. 定義運算子堆疊 operatorStack 和運算元堆疊 operandStack

  2. 從左至右掃描運算式,遇到運算元時,直接將其推入運算元堆疊 operandStack

  3. 遇到運算子時,比較其與運算子堆疊頂部運算子的優先級:

    • 如果該運算子的優先級高于或等于運算子堆疊頂部運算子,則將該運算子直接入堆疊 operatorStack

    • 如果該運算子的優先級低于運算子堆疊頂部運算子,則將運算子堆疊頂部的運算子出堆疊,從運算元堆疊中彈出兩個運算元,計算結果后再入堆疊 operandStack ,重復此步驟直到運算子堆疊為慷訓遇到優先級高于或等于該運算子的堆疊頂運算子為止,

  4. 遇到括號時:

    • 如果是左括號“(”,則直接入堆疊 operatorStack

    • 如果是右括號“)”,則將運算子堆疊堆疊頂的運算子出堆疊,從運算元堆疊中彈出兩個運算元計算結果,重復此步驟直到遇到左括號為止,并將這一對括號從運算子堆疊中移除,

  5. 重復步驟3和4,直到運算式的最右端,

  6. 將運算子堆疊中剩余的所有運算子依次出堆疊,從運算元堆疊中彈出兩個運算元,計算結果后入堆疊operandStack,

  7. 運算元堆疊最終只剩一個運算元,這就是運算式的計算結果,

具體實作代碼如下:

class ExpressionEvaluator
{
    static Dictionary<char, int> PrecedenceDic = new Dictionary<char, int> {
            {'+', 1}, {'-', 1}, {'*', 2}, {'/', 2}, {'^', 3}
        };

    static Dictionary<char, Func<int, int, int>> OperatorsDic = new Dictionary<char, Func<int, int, int>> {
            {'+', (a, b) => a + b },
            {'-', (a, b) => a - b },
            {'*', (a, b) => a * b },
            {'/', (a, b) => a / b },
            {'^', (a, b) => (int)Math.Pow(a, b)}
        };

    public static bool EvaluateExpression(string expression, out double result)
    {
        result = 0;
        try
        {
            // 使用正則運算式驗證四則運算運算式的有效性
            string pattern = @"^[-+*/^() x\d\s]+$";

            if (!Regex.IsMatch(expression, pattern))
            {
                return false;
            }
            //運算子堆疊
            Stack<char> operatorStack = new Stack<char>();
            //運算元堆疊
            Stack<int> operandStack = new Stack<int>();

            for (int i = 0; i < expression.Length; i++)
            {
                char c = expression[i];

                if (c == ' ') continue;

                if (char.IsDigit(c))
                {
                    //獲取運算元
                    int operand = 0;
                    while (i < expression.Length && char.IsDigit(expression[i]))
                    {
                        operand = operand * 10 + (expression[i++] - '0');
                    }
                    i--;
                    operandStack.Push(operand);
                }
                else if (OperatorsDic.ContainsKey(c))
                {
                    while (operatorStack.Count > 0 &&
                        OperatorsDic[c] != null && operatorStack.Peek() != '(' &&
                        PrecedenceDic[operatorStack.Peek()] >= PrecedenceDic[c])
                    {
                        int b = operandStack.Pop();
                        int a = operandStack.Pop();
                        operandStack.Push(OperatorsDic[operatorStack.Pop()](a, b));
                    }
                    operatorStack.Push(c);
                }
                else if (c == '(')
                {
                    operatorStack.Push(c);
                }
                else if (c == ')')
                {
                    while (operatorStack.Peek() != '(')
                    {
                        int b = operandStack.Pop();
                        int a = operandStack.Pop();
                        operandStack.Push(OperatorsDic[operatorStack.Pop()](a, b));
                    }
                    operatorStack.Pop();
                }
            }

            while (operatorStack.Count > 0)
            {
                int b = operandStack.Pop();
                int a = operandStack.Pop();
                operandStack.Push(OperatorsDic[operatorStack.Pop()](a, b));
            }
            result = operandStack.Pop();

            return true;
        }
        catch (Exception)
        {
            return false;
        }
    }
}

那接下來測驗一下代碼,因為代碼內只做了整形的計算,所以運算式也只用整形,

image.png

官方API

實際上微軟官方在 System.Data 庫中 DataTable.Compute(String, String)方法實作了計算運算式,代碼如下

using System;
using System.Data;
using System.Text.RegularExpressions;

public class ArithmeticExpressionEvaluator
{
    public static bool IsArithmeticExpression(int arg, string str, out double result)
    {
        result = 0;

        // 驗證字串是否包含有效的四則運算運算式
        if (!IsValidArithmeticExpression(str) || !str.ToLower().Contains("x".ToLower()))
        {
            return false;
        }

        // 將字串中的變數x替換為傳入的整數arg
        string expression = str.Replace("x", arg.ToString());

        // 計算并回傳運算式的值
        try
        {
            return double.TryParse(new DataTable().Compute(expression, "").ToString(), out result);
        }
        catch
        {
            return false;
        }
    }

    private static bool IsValidArithmeticExpression(string str)
    {
        // 使用正則運算式驗證四則運算運算式的有效性
        string pattern = @"^[-+*/() x\d\s]+$";
        return Regex.IsMatch(str, pattern);
    }
}
class Program
{
    public static void Main()
    {
        while (true)
        {
            string expression = Console.ReadLine();
            string arg = Console.ReadLine();

            if (ArithmeticExpressionEvaluator.IsArithmeticExpression(int.Parse(arg), expression, out double result))
            {
                Console.WriteLine($"The result of the arithmetic expression is: {result}");
            }
            else
            {
                Console.WriteLine("The input string is not a valid arithmetic expression.");
            }
        }
    }
}

測驗結果:

image.png

總結

剛開始拿到這個需求還是有點頭疼的,想了很久的方案,突然想到之前看資料結構的書的時候,提到過堆疊在運算式求值中的應用,翻書看了一下,還是被這個實作方案驚艷到了,所以,還是需要多讀多看多思考,才能在面對各種需求游刃有余,加油~

作者: Peter.Pan

出處: https://www.cnblogs.com/pandefu/>

郵箱: [email protected]

關于作者:.Net Framework,.Net Core ,WindowsForm,WPF ,控制元件庫,多執行緒

本文著作權歸作者和博客園共有,歡迎轉載,但未經作者同意必須保留此段宣告,且在文章頁面明顯位置給出 原文鏈接,否則保留追究法律責任的權利, 如有問題, 可郵件咨詢,

轉載請註明出處,本文鏈接:https://www.uj5u.com/net/557090.html

標籤:C#

上一篇:玻璃窯爐入爐配料稱重從硬體模型搭建到稱重資料寫表

下一篇:返回列表

標籤雲
其他(162435) Python(38274) JavaScript(25531) Java(18294) C(15240) 區塊鏈(8275) C#(7972) AI(7469) 爪哇(7425) MySQL(7296) html(6777) 基礎類(6313) sql(6102) 熊猫(6058) PHP(5876) 数组(5741) R(5409) Linux(5347) 反应(5209) 腳本語言(PerlPython)(5129) 非技術區(4971) Android(4616) 数据框(4311) css(4259) 节点.js(4032) C語言(3288) json(3245) 列表(3129) 扑(3119) C++語言(3117) 安卓(2998) 打字稿(2995) VBA(2789) Java相關(2746) 疑難問題(2699) 细绳(2522) 單片機工控(2479) iOS(2439) ASP.NET(2404) MongoDB(2323) 麻木的(2285) 正则表达式(2254) 字典(2211) 循环(2198) 迅速(2185) 擅长(2169) 镖(2155) HtmlCss(1998) .NET技术(1986) 功能(1967) Web開發(1951) C++(1942) python-3.x(1918) 弹簧靴(1913) xml(1889) PostgreSQL(1883) .NETCore(1863) 谷歌表格(1846) Unity3D(1843) for循环(1842)

熱門瀏覽
  • WebAPI簡介

    Web體系結構: 有三個核心:資源(resource),URL(統一資源識別符號)和表示 他們的關系是這樣的:一個資源由一個URL進行標識,HTTP客戶端使用URL定位資源,表示是從資源回傳資料,媒體型別是資源回傳的資料格式。 接下來我們說下HTTP. HTTP協議的系統是一種無狀態的方式,使用請求/ ......

    uj5u.com 2020-09-09 22:07:47 more
  • asp.net core 3.1 入口:Program.cs中的Main函式

    本文分析Program.cs 中Main()函式中代碼的運行順序分析asp.net core程式的啟動,重點不是剖析原始碼,而是理清程式開始時執行的順序。到呼叫了哪些實體,哪些法方。asp.net core 3.1 的程式入口在專案Program.cs檔案里,如下。ususing System; us ......

    uj5u.com 2020-09-09 22:07:49 more
  • asp.net網站作為websocket服務端的應用該如何寫

    最近被websocket的一個問題困擾了很久,有一個需求是在web網站中搭建websocket服務。客戶端通過網頁與服務器建立連接,然后服務器根據ip給客戶端網頁發送資訊。 其實,這個需求并不難,只是剛開始對websocket的內容不太了解。上網搜索了一下,有通過asp.net core 實作的、有 ......

    uj5u.com 2020-09-09 22:08:02 more
  • ASP.NET 開源匯入匯出庫Magicodes.IE Docker中使用

    Magicodes.IE在Docker中使用 更新歷史 2019.02.13 【Nuget】版本更新到2.0.2 【匯入】修復單列匯入的Bug,單元測驗“OneColumnImporter_Test”。問題見(https://github.com/dotnetcore/Magicodes.IE/is ......

    uj5u.com 2020-09-09 22:08:05 more
  • 在webform中使用ajax

    如果你用過Asp.net webform, 說明你也算是.NET 開發的老兵了。WEBform應該是2011 2013左右,當時還用visual studio 2005、 visual studio 2008。后來基本都用的是MVC。 如果是新開發的專案,估計沒人會用webform技術。但是有些舊版 ......

    uj5u.com 2020-09-09 22:08:50 more
  • iis添加asp.net網站,訪問提示:由于擴展配置問題而無法提供您請求的

    今天在iis服務器配置asp.net網站,遇到一個問題,記錄一下: 問題:由于擴展配置問題而無法提供您請求的頁面。如果該頁面是腳本,請添加處理程式。如果應下載檔案,請添加 MIME 映射。 WindowServer2012服務器,添加角色安裝完.netframework和iis之后,運行aspx頁面 ......

    uj5u.com 2020-09-09 22:10:00 more
  • WebAPI-處理架構

    帶著問題去思考,大家好! 問題1:HTTP請求和回傳相應的HTTP回應資訊之間發生了什么? 1:首先是最底層,托管層,位于WebAPI和底層HTTP堆疊之間 2:其次是 訊息處理程式管道層,這里比如日志和快取。OWIN的參考是將訊息處理程式管道的一些功能下移到堆疊下端的OWIN中間件了。 3:控制器處理 ......

    uj5u.com 2020-09-09 22:11:13 more
  • 微信門戶開發框架-使用指導說明書

    微信門戶應用管理系統,采用基于 MVC + Bootstrap + Ajax + Enterprise Library的技術路線,界面層采用Boostrap + Metronic組合的前端框架,資料訪問層支持Oracle、SQLServer、MySQL、PostgreSQL等資料庫。框架以MVC5,... ......

    uj5u.com 2020-09-09 22:15:18 more
  • WebAPI-HTTP編程模型

    帶著問題去思考,大家好!它是什么?它包含什么?它能干什么? 訊息 HTTP編程模型的核心就是訊息抽象,表示為:HttPRequestMessage,HttpResponseMessage.用于客戶端和服務端之間交換請求和回應訊息。 HttpMethod類包含了一組靜態屬性: private stat ......

    uj5u.com 2020-09-09 22:15:23 more
  • 部署WebApi隨筆

    一、跨域 NuGet參考Microsoft.AspNet.WebApi.Cors WebApiConfig.cs中配置: // Web API 配置和服務 config.EnableCors(new EnableCorsAttribute("*", "*", "*")); 二、清除默認回傳XML格式 ......

    uj5u.com 2020-09-09 22:15:48 more
最新发布
  • 編碼技巧 --- 如何實作字串運算運算式的計算

    ## 引言 最近做一個配置的功能,需求是該配置項跟另一個整形配置項關聯,具有一定的函式關系,例如有一個配置項是值為 `N` ,則另一配置 `F` 項滿足函式關系$F=2/(N+1)$。這個函式關系是客戶手動輸入,只需要簡單的四則運算,所以我們要做的就是判斷四則運算運算式是否有效,且給定 `N` 的值 ......

    uj5u.com 2023-07-12 08:55:12 more
  • 玻璃窯爐入爐配料稱重從硬體模型搭建到稱重資料寫表

    北辰模塊 IPv4 地址: 10.30.15.244IPv4 子網掩碼: 255.255.255.0 IPv4 默認網關: 10.30.15.254 串口服務器 ......

    uj5u.com 2023-07-07 09:10:05 more
  • 玻璃窯爐入爐配料稱重從硬體模型搭建到稱重資料寫表

    北辰模塊 IPv4 地址: 10.30.15.244IPv4 子網掩碼: 255.255.255.0 IPv4 默認網關: 10.30.15.254 串口服務器 ......

    uj5u.com 2023-07-07 09:09:34 more
  • Prism導航

    > 通常,導航意味著某個Control被添加到UI中,與此同時另一個Control被移除。 # 簡單跳轉 1. 新建 `UserControl`,新建ViewModel,VM需要實作 `INavigationAware` 2. 注冊 `UserControl`到DryIoc容器 ``` contai ......

    uj5u.com 2023-06-26 09:37:26 more
  • Prism導航

    > 通常,導航意味著某個Control被添加到UI中,與此同時另一個Control被移除。 # 簡單跳轉 1. 新建 `UserControl`,新建ViewModel,VM需要實作 `INavigationAware` 2. 注冊 `UserControl`到DryIoc容器 ``` contai ......

    uj5u.com 2023-06-26 09:36:38 more
  • .netcore中的虛擬檔案EmbeddedFile

    以前一直比較好奇像swagger,cap,skywalking等組件是如何實作參考一個dll即可在網頁上展示界面的,難道這么多html,js,css等都是硬編碼寫死在代碼檔案中的?后面接觸apb里面也有虛擬檔案的功能,一直沒去深入了解,最近仔細看了一下他們的代碼,發現內部其實就是用嵌入式檔案(Emb ......

    uj5u.com 2023-06-08 09:47:43 more
  • .netcore中的虛擬檔案EmbeddedFile

    以前一直比較好奇像swagger,cap,skywalking等組件是如何實作參考一個dll即可在網頁上展示界面的,難道這么多html,js,css等都是硬編碼寫死在代碼檔案中的?后面接觸apb里面也有虛擬檔案的功能,一直沒去深入了解,最近仔細看了一下他們的代碼,發現內部其實就是用嵌入式檔案(Emb ......

    uj5u.com 2023-06-08 09:46:11 more
  • C# 客戶端程式 Visual Studio 遠程除錯方法

    <a href="https://www.cnblogs.com/BoiledYakult/" target="_blank"><img width="48" height="48" class="pfs" src="https://pic.cnblogs.com/face/2913706/20230129104342.png" alt="" /&g...

    uj5u.com 2023-06-06 13:44:54 more
  • C# 版本特性一覽

    <a href="https://www.cnblogs.com/gaoyunpeng/" target="_blank"><img width="48" height="48" class="pfs" src="https://pic.cnblogs.com/face/85624/20130508133555.png" alt="" />&l...

    uj5u.com 2023-06-06 13:30:51 more
  • 基于SqlSugar的開發框架循序漸進介紹(31)-- 在查詢介面中實作多表

    在一些復雜的業務表中間查詢資料,有時候操作會比較復雜一些,不過基于SqlSugar的相關操作,處理的代碼會比較簡單一些,以前我在隨筆《基于SqlSugar的開發框架循序漸進介紹(2)-- 基于中間表的查詢處理》介紹過基于主表和中間表的聯合查詢,而往往實際會比這個會復雜一些。本篇隨筆介紹聯合多個表進行... ......

    uj5u.com 2023-06-03 08:56:54 more