計算數學算式的值

来源:https://www.cnblogs.com/lwcdi/archive/2023/06/27/17508214.html
-Advertisement-
Play Games

## 一:背景 ### 1. 講故事 前段時間有個朋友找到我,說他們的程式有偶發崩潰的情況,讓我幫忙看下怎麼回事,針對這種 crash 的程式,用 AEDebug 的方式抓取一個便知,有了 dump 之後接下來就可以分析了。 ## 二:Windbg 分析 ### 1. 為什麼會崩潰 既然是程式的崩潰 ...


實現語言:C#

 

思路:

1.括弧裡面的計算優先順序最高,首先計算出括弧裡面的值;

2.將所有的括弧值計算完成後,將得到不包含括弧的數學算式A;

3.數學算式A按先乘除後加減的優先順序計算處結果。

 

使用單的技術:

在解析數據算式過程中使用棧結構,可以比較方便比配左右括弧。

 

/// <summary>

/// 計算表達式的值

/// 例子:3/3-[(1/2)+(2*3)]*(1+2)*100+100

/// </summary>

/// <param name="expressionInput"></param>

/// <returns></returns>

public static decimal CalculateExpression(this string expressionInput)

{

    try

    {

        string expression = expressionInput.RemoveNewLineAndSpace();

        Stack<char> stack = new Stack<char>(1000);

        List<char> startChars = new List<char>(4) { '(', '[', '{' };

        List<char> endChars = new List<char>(4) { ')', ']', '}' };

        foreach (char item in expression)

        {

            if (!endChars.Contains(item))

            {

                stack.Push(item);

                continue;

            }

            StringBuilder childExpression = new StringBuilder();

            while (true)

            {

                if (stack.Count < 1) break;

                char expressionContent = stack.Pop();

                if (startChars.Contains(expressionContent)) break;

                childExpression.Insert(0, expressionContent);

            }

            decimal tempResult = CalculateSimpleExpression(childExpression.ToString());

            //Replace("-","|") 處理負數運算

            tempResult.ToString().Replace("-", "|").ToCharArray().ToList().ForEach(c => stack.Push(c));

        }

        StringBuilder simpleExpression = new StringBuilder();

        while (stack.Count > 0)

        {

            simpleExpression.Insert(0, stack.Pop());

        }

        return CalculateSimpleExpression(simpleExpression.ToString());

    }

    catch

    {

        throw new Exception("計算表達式的值報錯");

    }

}

 

 

/// <summary>

/// 計算簡單表達式的值

/// 不能包含括弧

/// </summary>

/// <param name="exp"></param>

/// <returns></returns>

private static decimal CalculateSimpleExpression(string exp)

{

    if (decimal.TryParse(exp, out decimal r))

    {

        return r;

    }

    if (exp.StartsWith("-") || exp.StartsWith("+"))

    {

        exp = $"0{exp}";

    }

    List<char> operateCodes = new List<char>(4) { '+', '-', '*', '/' };

    List<string> items = new List<string>();

    StringBuilder itemStr = new StringBuilder();

    exp.ToArray().ToList().ForEach(item => {

        if (!operateCodes.Contains(item))

        {

            itemStr.Append(item);

            return;

        }

        items.Add(itemStr.ToString().Replace("|", "-"));//Replace("-","|") 處理負數運算

        items.Add(item.ToString());

        itemStr.Clear();

    });

    items.Add(itemStr.ToString().Replace("|", "-"));

 

    // originalValues:原始項  operates:操作符

    Func<List<string>, List<string>, Stack<string>> CalExpression = (originalValues, operates) =>

    {

        Stack<string> calculatingExpression = new Stack<string>(16);

        foreach (string item in originalValues)

        {

            bool existOperateCode = operates.Count(c => calculatingExpression.Contains(c)) > 0 ? true : false;

            if (existOperateCode == false)

            {

                calculatingExpression.Push(item);

                continue;

            }

            var cal = new Dictionary<string, Func<decimal, decimal, decimal>>();

            cal.Add("*", (a, b) => a * b);

            cal.Add("/", (a, b) => a / b);

            cal.Add("+", (a, b) => a + b);

            cal.Add("-", (a, b) => a - b);

            decimal tempValue = cal[calculatingExpression.Pop()](decimal.Parse(calculatingExpression.Pop()), decimal.Parse(item));

            calculatingExpression.Push(tempValue.ToString());

        }

        return calculatingExpression;

    };

    //先計算乘除

    items = CalExpression(items, new List<string>() { "*", "/" }).ToList();

    items.Reverse();

    //先計算加減

    var resultStack = CalExpression(items, new List<string>() { "+", "-" });

    return decimal.Parse(resultStack.Pop());

}

 

驗證:

"3/3-[(1/2)+(2*3)]*(1+2)*100+100".CalculateExpression();

結果:-1849

 


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

-Advertisement-
Play Games
更多相關文章
  • # 起因 我們在寫代碼時,有時間代碼過高,尤其是stream流的時間,可能有多個map,filter,sort組成,這樣我們更希望看到的是一種可讀性更好的風格 * 我的代碼是這樣的 ![](https://img2023.cnblogs.com/blog/118538/202306/118538-2 ...
  • 這個問題非常有趣,不是SpringMVC 的問題,是實際開發中混合使用了兩種請求方式暴露出來的。 ## 問題場景 功能模塊中,提供兩個 Http 服務。一個是列表查詢(application/json 請求),一個是列表導出(表單請求)。運行環境發現個問題:MVC model 新添加的屬性,類似的 ...
  • 本篇文章以結解決國內ip無法訪問chatgpt介面為切入點,熟悉並瞭解Vxray,掌握魔法小梯+Vxray配合操作,openai-gpt3-java 配合使用代理實現訪問 ...
  • std::future 是一個 C++11 引入的標準庫類,可用於非同步獲取計算結果。通常情況下,std::future 可以通過 get() 函數來等待非同步操作完成,並獲取其結果。 如果需要等待多個非同步操作完成並獲取它們各自的結果,可以使用 std::future 的姊妹類 std::shared_ ...
  • > 2023/6/27 通信之間的實現,目的是為了,方便回顧時立馬能使用。具體的比如什麼情況選用什麼通信,各自的優點,沒有記錄。原因是,本人目前實戰經驗較少,還處於學習狀態,對此的理解還停留在管道方便,共用記憶體效率高,控制操作用消息隊列 # 前提 1. 進程間通信多少種? 答:7種,分別為無名管道、 ...
  • 面對同樣的客戶端請求,SpringCloud Gateway可以轉發給不同的服務去處理,掌握這個技能,讓請求從微服務入口處被掌控,被調度 ...
  • 1 # -*- coding: utf-8 -*- 2 """ 3 @Time : 2023-06-25 23:14 4 @Description : 5 @Author : Mr.Gu 6 """ 7 import time 8 import types 9 10 11 class Meta(ty ...
  • .NET Core 6引入了創建單文件可執行文件的功能。這隻允許分發一個應用程式文件,因為所有配置和依賴項都包含在二進位文件本身中。 該功能為依賴項嵌入提供了一種本機方法,這在發佈生成數百個程式集的獨立應用程式時最有益。它可用於依賴於框架或自包含的應用程式,但在這兩種情況下都需要設置運行時標識符以針 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...