最近重新翻起以前学的数据结构,原来是用C写的,今天用C#重新回温一下,加深一点印象.
目前还不支持单目操作符,只支持双目的,以后改进吧.:)
using System;
using System.Drawing;
using System.Collections;
using System.ComponentModel;
using System.Windows.Forms;
using System.Data;
namespace Expression1
{
/// <summary>
/// Form1 的摘要说明。
/// </summary>
public class Form1 : System.Windows.Forms.Form
{
private System.Windows.Forms.Label label1;
private System.Windows.Forms.Label label2;
private System.Windows.Forms.TextBox txtInfix;
private System.Windows.Forms.TextBox txtPostfix;
private System.Windows.Forms.Button btnPostfix;
private System.Windows.Forms.Button btnCalc;
private System.Windows.Forms.TextBox txtCalc;
/// <summary>
/// 必需的设计器变量。
/// </summary>
private System.ComponentModel.Container components = null;
public Form1()
{
//
// Windows 窗体设计器支持所必需的
//
InitializeComponent();
//
// TODO: 在 InitializeComponent 调用后添加任何构造函数代码
//
}
/// <summary>
/// 清理所有正在使用的资源。
/// </summary>
protected override void Dispose( bool disposing )
{
if( disposing )
{
if (components != null)
{
components.Dispose();
}
}
base.Dispose( disposing );
}
#region Windows 窗体设计器生成的代码
/// <summary>
/// 设计器支持所需的方法 - 不要使用代码编辑器修改
/// 此方法的内容。
/// </summary>
private void InitializeComponent()
{
this.txtInfix = new System.Windows.Forms.TextBox();
this.txtPostfix = new System.Windows.Forms.TextBox();
this.btnPostfix = new System.Windows.Forms.Button();
this.label1 = new System.Windows.Forms.Label();
this.label2 = new System.Windows.Forms.Label();
this.btnCalc = new System.Windows.Forms.Button();
this.txtCalc = new System.Windows.Forms.TextBox();
this.SuspendLayout();
//
// txtInfix
//
this.txtInfix.Location = new System.Drawing.Point(88, 24);
this.txtInfix.Name = "txtInfix";
this.txtInfix.Size = new System.Drawing.Size(176, 21);
this.txtInfix.TabIndex = 0;
this.txtInfix.Text = "";
//
// txtPostfix
//
this.txtPostfix.Location = new System.Drawing.Point(88, 88);
this.txtPostfix.Name = "txtPostfix";
this.txtPostfix.Size = new System.Drawing.Size(176, 21);
this.txtPostfix.TabIndex = 1;
this.txtPostfix.Text = "";
//
// btnPostfix
//
this.btnPostfix.FlatStyle = System.Windows.Forms.FlatStyle.System;
this.btnPostfix.Location = new System.Drawing.Point(120, 56);
this.btnPostfix.Name = "btnPostfix";
this.btnPostfix.TabIndex = 2;
this.btnPostfix.Text = "转化";
this.btnPostfix.Click += new System.EventHandler(this.btnPostfix_Click);
//
// label1
//
this.label1.Location = new System.Drawing.Point(8, 24);
this.label1.Name = "label1";
this.label1.Size = new System.Drawing.Size(72, 23);
this.label1.TabIndex = 3;
this.label1.Text = "中缀表达式";
this.label1.TextAlign = System.Drawing.ContentAlignment.MiddleCenter;
//
// label2
//
this.label2.Location = new System.Drawing.Point(8, 88);
this.label2.Name = "label2";
this.label2.Size = new System.Drawing.Size(72, 23);
this.label2.TabIndex = 4;
this.label2.Text = "后缀表达式";
this.label2.TextAlign = System.Drawing.ContentAlignment.MiddleCenter;
//
// btnCalc
//
this.btnCalc.FlatStyle = System.Windows.Forms.FlatStyle.System;
this.btnCalc.Location = new System.Drawing.Point(24, 136);
this.btnCalc.Name = "btnCalc";
this.btnCalc.TabIndex = 5;
this.btnCalc.Text = "计算结果";
this.btnCalc.Click += new System.EventHandler(this.btnCalc_Click);
//
// txtCalc
//
this.txtCalc.Location = new System.Drawing.Point(120, 136);
this.txtCalc.Name = "txtCalc";
this.txtCalc.Size = new System.Drawing.Size(144, 21);
this.txtCalc.TabIndex = 6;
this.txtCalc.Text = "";
//
// Form1
//
this.AutoScaleBaseSize = new System.Drawing.Size(6, 14);
this.ClientSize = new System.Drawing.Size(292, 182);
this.Controls.Add(this.txtCalc);
this.Controls.Add(this.btnCalc);
this.Controls.Add(this.label2);
this.Controls.Add(this.label1);
this.Controls.Add(this.btnPostfix);
this.Controls.Add(this.txtPostfix);
this.Controls.Add(this.txtInfix);
this.MaximizeBox = false;
this.Name = "Form1";
this.Text = "表达式计算";
this.ResumeLayout(false);
}
#endregion
/// <summary>
/// 应用程序的主入口点。
/// </summary>
[STAThread]
static void Main()
{
Application.EnableVisualStyles();
Application.DoEvents();
Application.Run(new Form1());
}
private void btnPostfix_Click(object sender, System.EventArgs e)
{
try
{
Calculator calc = new Calculator(this.txtInfix.Text.Replace(" ",""));
this.txtPostfix.Text = calc.PostFix();
}
catch(Expressception ex)
{
MessageBox.Show(ex.Message);
}
}
private void btnCalc_Click(object sender, System.EventArgs e)
{
try
{
Calculator calc = new Calculator(this.txtInfix.Text.Replace(" ",""));
this.txtCalc.Text = calc.Run().ToString();
}
catch(Expressception ex)
{
MessageBox.Show(ex.Message);
}
}
}
public class Calculator
{
private string _expression;
private Stack s;
public Calculator(string expression)
{
this._expression = expression;
s = new Stack();
}
public double Run()
{
string expression=PostFix();
string[] aryString = expression.Split('|');
foreach(string str in aryString)
{
if(IsNumber(str))
{
double d = Convert.ToDouble(str.ToString());
AddOperands(d);
}
else
{
DoOperator(str);
}
}
return (double)s.Pop();
}
private bool IsNumber(string str)
{
if(str.Length>1)
{
return true;
}
else
{
return Char.IsDigit(str[0]);
}
}
private void AddOperands(double val)
{
s.Push(val);
}
private bool Get2Operands(out double left,out double right)
{
try
{
right = (double)s.Pop();
left = (double)s.Pop();
}
catch(InvalidOperationException)
{
right = 0;
left = 0;
return false;
}
return true;
}
private void DoOperator(string op)
{
double left,right;
bool result = Get2Operands(out left,out right);
if(result)
switch(op)
{
case "+": s.Push(left+right); break;
case "-": s.Push(left-right); break;
case "*": s.Push(left*right); break;
case "/":
if(right == 0.0)
{
s.Clear();
//Divide by 0!
throw new Expressception("除数不能为零");
}
else
s.Push(left/right);
break;
case "^":
s.Push(Math.Pow(left,right));
break;
}
else
s.Clear();
}
public string PostFix()
{
string str = this._expression+"#";
char tempc;
char[] chars = str.ToCharArray();
Stack ts = new Stack();
ts.Push('#');
string str1="";
string tmpStr="";
bool isNum = false;
foreach(char c in chars)
{
if(Char.IsDigit(c)||c=='.')
{
tmpStr += c.ToString();
isNum = true;
}
else
{
if(isNum)
{
str1 += tmpStr+"|";
tmpStr = "";
}
isNum=false;
if(c == ')')
{
for(tempc = Convert.ToChar(ts.Pop());tempc!='(';tempc = Convert.ToChar(ts.Pop()))
str1+=tempc.ToString()+"|";
}
else
{
for(tempc = Convert.ToChar(ts.Pop());Isp(tempc)>Icp(c);tempc = Convert.ToChar(ts.Pop()))
str1+=tempc.ToString()+"|";
ts.Push(tempc);
ts.Push(c);
}
}
}
return str1.Substring(0,str1.Length-1);
}
private int Isp(char c)
{
int k;
switch(c)
{
case '#': k=0; break;
case '(': k=1; break;
case '^': k=7; break;
case '*':
case '/':
case '%': k=5; break;
case '+':
case '-': k=3; break;
case ')': k=8; break;
default:
//Unknown operator!
throw new Expressception("不知道的操作符!");
}
return k;
}
private int Icp(char c)
{
int k;
switch(c)
{
case '#': k=0; break;
case '(': k=8; break;
case '^': k=6; break;
case '*':
case '/':
case '%': k=4; break;
case '+':
case '-': k=2; break;
case ')': k=1; break;
default:
//Unknown operator!
throw new Expressception("不知道的操作符!");
}
return k;
}
}
public class Expressception : Exception
{
public Expressception(string msg) : base(msg)
{
}
}