1.具体步骤

1)初始化两个栈:运算符栈s1和储存中间结果的栈s2;
2)从左至右扫描中缀表达式;
3)遇到操作数时,将其压s2;
4)遇到运算符时,比较其与s1栈顶运算符的优先级:
     (1)如果s1为空,或栈顶运算符为左括号“(”,则直接将此运算符入栈;
     (2)否则,若优先级比栈顶运算符的高,也将运算符压入s1;
     (3)否则,将s1栈顶的运算符弹出并压入到s2中,再次转到(4-1)与s1中新的栈顶运算符相比较;
5)遇到括号时:
     (1)如果是左括号“(”,则直接压入s1
     (2)如果是右括号“)”,则依次弹出s1栈顶的运算符,并压入s2,直到遇到左括号为止,此时将这一对括号丢弃
6)重复步骤2至5,直到表达式的最右边
7)将s1中剩余的运算符依次弹出并压入s2
8)依次弹出s2中的元素并输出,结果的逆序即为中缀表达式对应的后缀表达式

2.思路分析

3.代码实现

 /// <summary>
/// 字符串转中缀表达式的List
/// </summary>
/// <param name="expression"></param>
/// <returns></returns>
public static List<string> ToInfixExpression(string expression)
{
List<string> list = new List<string>(); int index = ; string str = ""; do
{ //48-57ASCII码代表的是0-9 如果不是0-9直接入链表
if (expression[index] < || expression[index] > )//ascii编码
{
list.Add("" + expression[index]); index++;
}
else
{
str = ""; //多位数判断
while (index < expression.Length && expression[index] >= && expression[index] <= )
{ str += expression[index]; index++;
} list.Add(str);
} } while (index < expression.Length); return list;
}
    /// <summary>
/// 中缀转后缀
/// </summary>
/// <param name="expression"></param>
/// <returns></returns>
public static List<string> ParseSuffixExpression(List<string> expression)
{
//存储中间结果
List<string> list = new List<string>();
//符号栈
Stack<string> stack = new Stack<string>(); foreach (var item in expression)
{
//多位数判断 如果是数字直接加入list
if (Regex.IsMatch(item, "\\d+"))
{
list.Add(item);
}
//如果是左括号,直接入符号栈
else if (item.Equals("("))
{
stack.Push(item);
}
//如果是右括号
else if (item.Equals(")"))
{
//依次弹出stack栈顶的运算符,并存入list,直到遇到左括号为止
while (!stack.Peek().Equals("("))
{
list.Add(stack.Pop());
}
//将(也出栈
stack.Pop();
}
//如果是*/+-
else
{
//循环判断item的优先级小于或者等于stack栈顶运算符,将stack栈顶的运算符出栈并加入到list中
while (stack.Count != && Operation.GetValue(stack.Peek()) >= Operation.GetValue(item))
{
list.Add(stack.Pop());
}
//将item入栈
stack.Push(item);
}
} //将stack剩余的运算符依次入list
while (stack.Count!=)
{
list.Add(stack.Pop());
} return list;
}
    public class Operation
{
private static int ADD = ;
private static int SUB = ;
private static int MUL = ;
private static int DIV = ; public static int GetValue(string operation)
{
int result = ; switch (operation)
{
case "+":
result = ADD;
break;
case "-":
result = SUB;
break;
case "*":
result = MUL;
break;
case "/":
result = DIV;
break;
default:
// Console.WriteLine("不存在该运算符");
break;
} return result;
}
}
        /// <summary>
/// 计算
/// </summary>
/// <param name="list"></param>
/// <returns></returns>
public static int Calculate(List<string> list)
{
//创建栈
Stack<string> stack = new Stack<string>(); //循环遍历
list.ForEach(item =>
{
//正则表达式判断是否是数字,匹配的是多位数
if (Regex.IsMatch(item,"\\d+"))
{
//如果是数字直接入栈
stack.Push(item);
}
//如果是操作符
else
{
//出栈两个数字,并运算,再入栈
int num1 =int.Parse(stack.Pop()); int num2 = int.Parse(stack.Pop()); int result = ; if(item.Equals("+"))
{
result = num2 + num1;
}
else if(item.Equals("*"))
{
result = num2 * num1;
}
else if(item.Equals("/"))
{
result = num2 / num1;
}
else if (item.Equals("-"))
{
result = num2 - num1;
}
else
{
throw new Exception("无法识别符号");
} stack.Push(""+result);
}
}); //最后把stack中数据返回
return int.Parse(stack.Pop());
}
    public class ReversePolandTransformation
{ public static void Test()
{
string expression = "1+((2+3)*4)-5"; //将字符串转换成List
List<string> infixExpression = ToInfixExpression(expression); string str = ""; infixExpression.ForEach(item =>
{
str = str + item + ",";
}); Console.WriteLine("中缀表达式对应的List:"+str); str = ""; //将中缀表达式转换成后缀表达式
List<string> suffixExpression = ParseSuffixExpression(infixExpression); suffixExpression.ForEach(item =>
{
str = str + item + ",";
}); Console.WriteLine("\n后缀表达式对应的List:"+str); //结果计算
int result =PolandNotation.Calculate(suffixExpression); Console.WriteLine($"\n{expression}={result}");
}
}

最新文章

  1. RCP:如何移除Search对话框中不需要的项
  2. 计算机管理打不开的解决方法,直接cmd修改reg
  3. 面试体验:Google 篇(转)
  4. iOS开发 Xcode中的Info.plist字段含义
  5. .net I/O操作 导图
  6. 半个月学习的it内容
  7. MTV 和 MVC
  8. Java设计模式之动态代理
  9. Android为TV端助力 MediaPlayer的一些使用方法简历
  10. python变量类型&amp;字符串的内建函数使用
  11. Tell Me About Yourself Example 1
  12. linux及安全第五周总结
  13. AGC027 D - Modulo Matrix 构造
  14. npm安装时一些错误
  15. formidable模块的使用
  16. [uboot]在uboot里面添加环境变量使用run来执行
  17. 20155319 2016-2017-2 《Java程序设计》第六周学习总结
  18. 一图看懂hadoop Yarn工作原理
  19. BZOJ2257 [Jsoi2009]瓶子和燃料 【裴蜀定理】
  20. 关于ListView的注意点

热门文章

  1. 【QT】QT资料集锦
  2. [SD心灵鸡汤]007.每月一则 - 2015.11
  3. [安卓基础] 007.管理Activity的生命周期
  4. Mysql面试的技术名词
  5. 高性能可扩展mysql 笔记(四)项目分区表演示
  6. 【大厂面试03期】MySQL是怎么解决幻读问题的?
  7. Java实现 LeetCode 834 树中距离之和(DFS+分析)
  8. Java实现 LeetCode 757 设置交集大小至少为2(排序+滑动窗口)
  9. java实现孪生素数
  10. STL常用序列容器