关于表达式中除数为0的容错机制的处理

2024-01-23 06:50

本文主要是介绍关于表达式中除数为0的容错机制的处理,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

说明:对于一个表达式(比如1+2/(A-B)),如果A-B为0,有时我们想这个表达式返回1,而不是0或者抛出异常。也就是局部的除数异常是允许的。这个时候就需要对公式做容错处理。

下面是用C#的处理过程,如有错误,欢迎指正

/// <summary>/// 公式容错/// 在执行公式时,会有除数为0的情况,此时不应该报错,只是包含除数为0的表达式计算结果为0,整个表达式继续执行/// 如1+(2/(4-8/(1+1)))  执行结果应该是1 而不应该报错/// 方法:/// 1:将原表达式转换为后缀表达式/// 2:再将后缀表达式转换为容错之后的中缀表达式/// 3:栈顶元素即为结果/// </summary>public class FormulasConvert {//存原始后缀表达式Stack<char> _suffixStack = new Stack<char>();/// <summary>/// 后缀表达式结果/// </summary>public string _suffixFormula = string.Empty;//存容错后中缀表达式Stack<string> _infixingStack = new Stack<string>();/// <summary>/// 容错后中缀表达式结果/// </summary>public string _infixingFormula = string.Empty;/// <summary>/// 原始公式/// </summary>private string _inputFormula = string.Empty;/// <summary>/// 用字符转换之后的原始公式/// 10+10==》A+B/// </summary>private string _newFormula = string.Empty;//操作符private string _operator = "+-*/()";/// <summary>/// 原公式:100+2/300/// 对应:/// A:100  B:2 C:300/// 最终/// A+B/C/// </summary>Dictionary<string, string> _dicCompary = new Dictionary<string, string>();public FormulasConvert(string inputFormula) {this._inputFormula = inputFormula;}/// <summary>/// 将原始表达式转换成后缀表达式/// </summary>/// <param name="inputString">原中缀表达式</param>/// <returns></returns>public string ConvertToSuffix() {OperatorFormula();for (int i = 0; i < _newFormula.Length; i++) {char ch = _newFormula[i];if (!IsOperator(ch)) {_suffixFormula += ch;} else if (ch == '(')_suffixStack.Push(ch);else if (ch == ')') {while (_suffixStack.Peek() != '(')_suffixFormula += _suffixStack.Pop();_suffixStack.Pop();} else if (ch == '+' || ch == '-' || ch == '*' || ch == '/') {if (_suffixStack.Count == 0)_suffixStack.Push(ch);else if (_suffixStack.Peek() == '(')_suffixStack.Push(ch);else if (CompareOperate(ch, _suffixStack.Peek()) == 1)_suffixStack.Push(ch);else if (CompareOperate(ch, _suffixStack.Peek()) == 0) {_suffixFormula += _suffixStack.Pop();_suffixStack.Push(ch);} else if (CompareOperate(ch, _suffixStack.Peek()) == -1) {int com = -1;while (com == -1 || com == 0) {_suffixFormula += _suffixStack.Pop();if (_suffixStack.Count != 0)com = CompareOperate(ch, _suffixStack.Peek());elsebreak;}_suffixStack.Push(ch);}}}string tmpStr = string.Empty;for (int i = 0; i < _suffixStack.Count + 1; i++) {if (_suffixStack.Count > 0) {tmpStr += _suffixStack.Peek().ToString();_suffixStack.Pop();}}_suffixFormula = _suffixFormula + tmpStr;return GetFormulaResult(_suffixFormula);}private int CompareOperate(char ch, char stackCh) {if (ch == stackCh)return 0;else if ((ch == '+' && stackCh == '-') || (ch == '-' && stackCh == '+'))//表示等于return 0;else if ((ch == '*' && stackCh == '/') || (ch == '/' && stackCh == '*'))return 0;else if ((ch == '+' || ch == '-') && (stackCh == '*' || stackCh == '/'))//表示小于return -1;else if ((ch == '*' || ch == '/') && (stackCh == '+' || stackCh == '-'))//返回1 表示输入运算符的优先级大于栈顶运算符return 1;elsereturn -2;}/// <summary>/// 将后缀表达式生成中缀表达式(未容错)/// </summary>/// <returns></returns>public string ConvertToInfixing() {if (string.IsNullOrEmpty(this._inputFormula))return string.Empty;ConvertToSuffix();if (string.IsNullOrEmpty(this._suffixFormula))return string.Empty;for (int i = 0; i < _suffixFormula.Length; i++) {char ch = _suffixFormula[i];if (!IsOperator(ch)) { //操作数入栈_infixingStack.Push(ch.ToString());} else if (ch == '+' || ch == '-' || ch == '*' || ch == '/') {string tmp1 = _infixingStack.Pop();string tmp2 = _infixingStack.Pop();string tmp = string.Empty;tmp = "(" + tmp2;tmp += ch;tmp += tmp1 + ")";_infixingStack.Push(tmp);}}return GetFormulaResult(_infixingStack.Peek());}/// <summary>/// 将后缀表达式生成中缀表达式(容错,替换除数为0)/// 返回容错之后的新中缀表达式/// </summary>/// <param name="inputString"></param>/// <returns></returns>public string Convert() {if (string.IsNullOrEmpty(this._inputFormula))return string.Empty;ConvertToSuffix();if (string.IsNullOrEmpty(this._suffixFormula))return string.Empty;for (int i = 0; i < _suffixFormula.Length; i++) {char ch = _suffixFormula[i];if (!IsOperator(ch)) { //操作数入栈_infixingStack.Push(ch.ToString());} else if (ch == '+' || ch == '-' || ch == '*') {string tmp1 = _infixingStack.Pop();string tmp2 = _infixingStack.Pop();string tmp = string.Empty;tmp = "(" + tmp2;tmp += ch;tmp += tmp1 + ")";_infixingStack.Push(tmp);} else if (ch == '/') {//(A if(Ture) else B)string tmp1 = _infixingStack.Pop();string tmp2 = _infixingStack.Pop();string tmp = string.Empty;tmp = "(0 if(" + tmp1 + "==0) else (" + tmp2 + "/" + tmp1 + "))";_infixingStack.Push(tmp);}}return GetFormulaResult(_infixingStack.Peek());}/// <summary>/// 处理原始表达式为原运算符/// 最终变成A+B/C这种模式/// </summary>private void OperatorFormula() {if (string.IsNullOrEmpty(this._inputFormula))return;//26个字母应该够用了,一个公式有26个运算项已经够多了char start = 'A';string tmpData = string.Empty;for (int i = 0; i < this._inputFormula.Length; i++) {char ch = _inputFormula[i];if (IsOperator(ch)) {if (!string.IsNullOrEmpty(tmpData.Trim())) {this._dicCompary.Add(start.ToString(), tmpData);_newFormula += start.ToString();start = (char)((int)start + 1);tmpData = string.Empty;} else {tmpData = string.Empty;  //空格不处理}_newFormula += ch.ToString();} else {tmpData += _inputFormula[i].ToString();}}if (!string.IsNullOrEmpty(tmpData.Trim())) {_dicCompary.Add(start.ToString(), tmpData);_newFormula += start.ToString();  //加上最后一个操作数}}/// <summary>/// 根据_newFormula与_dicCompary重新组织成公式/// </summary>/// <returns></returns>private string GetFormulaResult(string formulas) {if (string.IsNullOrEmpty(formulas))return string.Empty;string newFormulas = string.Empty;for (int i = 0; i < formulas.Length; i++) {string key = formulas[i].ToString();if (_dicCompary.ContainsKey(key)) {newFormulas += _dicCompary[key];} else {newFormulas += key;}}return newFormulas;}//判断是否是操作符,目前操作符为+-*/()private bool IsOperator(char c) {return _operator.Contains(c.ToString());}}
测试:

private void button2_Click(object sender, EventArgs e) {FormulasConvert convert = new FormulasConvert(this.label2.Text);richTextBox2.Text = convert.ConvertToSuffix();}private void button3_Click(object sender, EventArgs e) {FormulasConvert convert = new FormulasConvert(this.label2.Text);richTextBox3.Text = convert.Convert();}

图例:



这篇关于关于表达式中除数为0的容错机制的处理的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/635556

相关文章

Python实现批量CSV转Excel的高性能处理方案

《Python实现批量CSV转Excel的高性能处理方案》在日常办公中,我们经常需要将CSV格式的数据转换为Excel文件,本文将介绍一个基于Python的高性能解决方案,感兴趣的小伙伴可以跟随小编一... 目录一、场景需求二、技术方案三、核心代码四、批量处理方案五、性能优化六、使用示例完整代码七、小结一、

Python中 try / except / else / finally 异常处理方法详解

《Python中try/except/else/finally异常处理方法详解》:本文主要介绍Python中try/except/else/finally异常处理方法的相关资料,涵... 目录1. 基本结构2. 各部分的作用tryexceptelsefinally3. 执行流程总结4. 常见用法(1)多个e

PHP应用中处理限流和API节流的最佳实践

《PHP应用中处理限流和API节流的最佳实践》限流和API节流对于确保Web应用程序的可靠性、安全性和可扩展性至关重要,本文将详细介绍PHP应用中处理限流和API节流的最佳实践,下面就来和小编一起学习... 目录限流的重要性在 php 中实施限流的最佳实践使用集中式存储进行状态管理(如 Redis)采用滑动

MyBatis-plus处理存储json数据过程

《MyBatis-plus处理存储json数据过程》文章介绍MyBatis-Plus3.4.21处理对象与集合的差异:对象可用内置Handler配合autoResultMap,集合需自定义处理器继承F... 目录1、如果是对象2、如果需要转换的是List集合总结对象和集合分两种情况处理,目前我用的MP的版本

Python自动化处理PDF文档的操作完整指南

《Python自动化处理PDF文档的操作完整指南》在办公自动化中,PDF文档处理是一项常见需求,本文将介绍如何使用Python实现PDF文档的自动化处理,感兴趣的小伙伴可以跟随小编一起学习一下... 目录使用pymupdf读写PDF文件基本概念安装pymupdf提取文本内容提取图像添加水印使用pdfplum

C# LiteDB处理时间序列数据的高性能解决方案

《C#LiteDB处理时间序列数据的高性能解决方案》LiteDB作为.NET生态下的轻量级嵌入式NoSQL数据库,一直是时间序列处理的优选方案,本文将为大家大家简单介绍一下LiteDB处理时间序列数... 目录为什么选择LiteDB处理时间序列数据第一章:LiteDB时间序列数据模型设计1.1 核心设计原则

基于Redis自动过期的流处理暂停机制

《基于Redis自动过期的流处理暂停机制》基于Redis自动过期的流处理暂停机制是一种高效、可靠且易于实现的解决方案,防止延时过大的数据影响实时处理自动恢复处理,以避免积压的数据影响实时性,下面就来详... 目录核心思路代码实现1. 初始化Redis连接和键前缀2. 接收数据时检查暂停状态3. 检测到延时过

Redis中哨兵机制和集群的区别及说明

《Redis中哨兵机制和集群的区别及说明》Redis哨兵通过主从复制实现高可用,适用于中小规模数据;集群采用分布式分片,支持动态扩展,适合大规模数据,哨兵管理简单但扩展性弱,集群性能更强但架构复杂,根... 目录一、架构设计与节点角色1. 哨兵机制(Sentinel)2. 集群(Cluster)二、数据分片

Java利用@SneakyThrows注解提升异常处理效率详解

《Java利用@SneakyThrows注解提升异常处理效率详解》这篇文章将深度剖析@SneakyThrows的原理,用法,适用场景以及隐藏的陷阱,看看它如何让Java异常处理效率飙升50%,感兴趣的... 目录前言一、检查型异常的“诅咒”:为什么Java开发者讨厌它1.1 检查型异常的痛点1.2 为什么说

Python利用PySpark和Kafka实现流处理引擎构建指南

《Python利用PySpark和Kafka实现流处理引擎构建指南》本文将深入解剖基于Python的实时处理黄金组合:Kafka(分布式消息队列)与PySpark(分布式计算引擎)的化学反应,并构建一... 目录引言:数据洪流时代的生存法则第一章 Kafka:数据世界的中央神经系统消息引擎核心设计哲学高吞吐