数据结构中用C#实现"表达式计算"

最近重新翻起以前学的数据结构,原来是用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)
  {
  }
 }

评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值