• 网络学院
  • 新手学堂
  • 操作系统
  • 网络技术
  • 软件应用
  • 办公软件
  • 编程技术
  • 网站架设
  • 数据库类
  • 平面设计
  • 多媒体类
  • 游戏资讯
  • 教学论文
  • 认证考试
用Visual
广告位
  站点:
  • 首 页
  • 最新更新
  • 软件分类
  • 国内软件
  • 国外软件
  • 汉化软件
  • 源码下载
  • 字体下载
用Visual C# 实现四则混合运算
软件发布 用Visual C# 实现四则混合运算
网络软件 系统工具 应用软件 联络聊天 图形图像 多媒体类 行业软件 游戏娱乐 编程开发 安全相关 教育教学 数码软件
热门软件: QQ 瑞星 pplive e话通 木马克星 千千静听 office2000 五笔字根 Photoshop 视频分割
返回首页 | 文章首页 | 编程技术 | C#教程 | 用Visual C# 实现四则混合运算

用Visual C# 实现四则混合运算

 

添加时间: 2007-7-16 3:33:37  作者: C#教程  阅读次数:136   来源: http://www.d9soft.com

 

 

        本文是介绍使用Visual C#实现自动计算四则混合运算的一种方法,在.net1.1框架中,还没有包含全现成的类,不过,现在经过我们下文的介绍,大家完全可以自己写一个,作用嘛,没什么,就是熟悉一下栈的用法,或者家里有上小学的,可以写一个程序给孩子练习一下四则混合运算也行,哈哈,废话不多说了,讲正题吧。

  出一个典型的算术题: (6+2*5)/4

  从题中,我们首先算 2*5=10 接下来算 6+10=16 最后算 16/4 =4 所以,结果是4

  计算机怎么算?还好前辈给我们列出来一堆的算法,我们随便选一个就可以了。

  第一种算法

  使用栈来解决,意思就是把不优极最低的压到栈的最下面去,按照先进后出的原则,那么最优先级最低的就是最后计算了。

  计算过程:

  我们建立两个栈,一个是数据栈,一个是计算符号栈,以(6+2*5)/4为例子,看看倒底是怎么计算的。

  假设:

  1)优先级

  符号之间的优先级如下:

  “(“ “)” -1

  “+”,”-” 0
 
   “*”,”/” 1

  数值越大,则越优先,同级别的比较时 先出现的优先。

  2)将”(”,”)”设为特殊运算符,即单目运算,两邻两个运算符则可对消。

  3) 计算条件

  (1) 当前运算符不等于“”(特殊结束符)

  (2) 运算符栈里的运行算个数>=1时

  (3) 出栈口的运算符优先级高于将要入栈的运算符时或者两者可对消时。

  计算时,则将符号出栈参与计算,数值栈的出栈口前两位元素出栈参与计算,计算结果值向数值栈压栈,并进行递归此操作。


图1:

  1) “(” 压入符号栈 2 ) “6”压入数值栈

  3) “(”与”+”比较优先级,认为”(”比”+”优先级低,则不满足计算条件,将”+”压入符号栈.


图2:

  1) 将”2” 压入数值栈。

  2) 将”*”与”+”比较优先级,算得”+”优先级低于”*”,则不满足计算条件,将”*”压入符号栈。


图3:

  1) 将 “5”压入数植栈。 2) 将“*“与”)”比较优先级,得出”*”比”)”优先级要高。进行计算,将”*”出栈、”5”、”2”出栈,参与计算


图4:

  1) 将 2*5 =10的结果压入数值栈。

  2) (递归)比较 “+”与”)”优先级,得出”+”比”)”优先级要高。再进行计算,将”+”出栈、”10”、”6”出栈,参与计算。


图 5:

  1) 将 6+10 =16的结果压入数值栈。

  2) (递归)比较 “)”与”(”优先级,得出两者可以对消,将”(”符号出栈,与”)”对消,继续取下一个符号。


图6:

  1) 将”/”入符号栈。

  2)将”4”入数值栈。

  3) 发现””算式结束符,则进行计算, 将 “/”、”4’、”16”出栈,参与计算。


图7:

  1) 将计算结果压入数值栈。

  成功了! 辛苦的计算工作终于干完了,看起来比人脑计算复杂多了:)

  第二种算法

  第二种方法,我们简单的提一下,在这里不作详细过程的叙述。第二种,就是使用树的方式,将一个算式组织成一定规律的树,之后,再进行遍历计算即得得到结果。还是以上面的算式作例子,最终形成的树的样式:(注意“()”这个符号要特殊处理)


图8:


  使用树的深度遍历即可算出最终的结果。

程序实现

  本文,给出使用栈处理四则运算的C#实现方法,如下:

CalUtility.cs

using System;
namespace Calculate
{
 /// <summary>
 /// CalUtility 的摘要说明。
 /// 读算式辅助工具
 /// </summary>

 public class CalUtility
 {
  System.Text.StringBuilder StrB;
  private int iCurr=0;
  private int iCount=0;

  /// <summary>
  /// 构造方法
  /// </summary>

  public CalUtility(string calStr)
  {
   StrB = new System.Text.StringBuilder(calStr.Trim());
   iCount = System.Text.Encoding.Default.GetByteCount(calStr.Trim());
  }

  /// <summary>
  /// 取段,自动分析数值或计算符
  /// </summary>
  /// <returns></returns>

  public string getItem()
  {
   //结束了
   if(iCurr==iCount)
    return "";
   char ChTmp = StrB[iCurr];
   bool b=IsNum(ChTmp);
   if(!b)
   {
    iCurr++;
    return ChTmp.ToString();
   }
   string strTmp="";
   while(IsNum(ChTmp)==b && iCurr<iCount)
   {
    ChTmp = StrB[iCurr];
    if(IsNum(ChTmp)==b)
     strTmp +=ChTmp;
    else
     break;
    iCurr++;
   }
   return strTmp;
  }

  /// <summary>
  /// 是否是数字
  /// </summary>
  /// <param name="c">内容</param>
  /// <returns></returns>

  public bool IsNum(char c)
  {
   if((c>=’0’ && c<=’9’) c==’.’)
   {
    return true;
   }
   else
   {
    return false;
   }
  }

  /// <summary>
  /// 是否是数字
  /// </summary>
  /// <param name="c">内容</param>
  /// <returns></returns>

  public bool IsNum(string c)
  {
   if(c.Equals(""))
    return false;
   if((c[0]>=’0’ && c[0]<=’9’) c[0]==’.’)
   {
    return true;
   }
   else
   {
    return false;
   }
  }

  /// <summary>
  /// 比较str1和str2两个运算符的优先级,ture表示str1高于str2,false表示str1低于str2
  /// </summary>
  /// <param name="str1">计算符1</param>
  /// <param name="str2">计算符2</param>
  /// <returns></returns>

  public bool Compare(string str1,string str2)
  {
   return getPriority(str1)>=getPriority(str2);
  }

  /// <summary>
  /// 取得计算符号的优先级
  /// </summary>
  /// <param name="str">计算符</param>
  /// <returns></returns>

  public int getPriority(string str)
  {
   if(str.Equals(""))
   {
    return -1;
   }
   if(str.Equals("("))
   {
    return 0;
   }
   if(str.Equals("+")str.Equals("-"))
   {
    return 1;
   }
   if(str.Equals("*")str.Equals("/"))
   {
    return 2;
   }
   if(str.Equals(")"))
   {
    return 0;
   }
   return 0;
  }
 }
}

IOper.cs
using System;

namespace Calculate
{
 /// <summary>
 /// IOper 的摘要说明。
 /// 计算符接口
 /// </summary>

 public interface IOper
 {
  /// <summary>
  /// 计算符计算接口计算方法
  /// </summary>
  /// <param name="o1">参数1</param>
  /// <param name="o2">参数2</param>
  /// <returns></returns>
  object Oper(object o1,object o2);
 }
}

Opers.cs
using System;

namespace Calculate
{
 /// <summary>
 /// Opers 的摘要说明。
 /// 各类计算符的接口实现,加减乘除
 /// </summary>

 public class OperAdd:IOper
 {
  public OperAdd()
  {
   //
   // TODO: 在此处添加构造函数逻辑
   //
  }
  #region IOper 成员

 public object Oper(object o1, object o2)
 {
  Decimal d1 = Decimal.Parse(o1.ToString());
  Decimal d2 = Decimal.Parse(o2.ToString());
  return d1+d2;
 }

 #endregion
}

public class OperDec:IOper
{
 public OperDec()
 {
  //
  // TODO: 在此处添加构造函数逻辑
  //
 }
 #region IOper 成员

 public object Oper(object o1, object o2)
 {
  Decimal d1 = Decimal.Parse(o1.ToString());
  Decimal d2 = Decimal.Parse(o2.ToString());
  return d1-d2;
 }
 #endregion
}

public class OperRide:IOper
{
 public OperRide()
 {
  //
  // TODO: 在此处添加构造函数逻辑
  //
 }
 #region IOper 成员

 public object Oper(object o1, object o2)
 {
  Decimal d1 = Decimal.Parse(o1.ToString());
  Decimal d2 = Decimal.Parse(o2.ToString());
  return d1*d2;
 }
 #endregion
}

public class OperDiv:IOper
{
 public OperDiv()
 {
  //
  // TODO: 在此处添加构造函数逻辑
  //
 }
 #region IOper 成员

 public object Oper(object o1, object o2)
 {
  Decimal d1 = Decimal.Parse(o1.ToString());
  Decimal d2 = Decimal.Parse(o2.ToString());
  return d1/d2;
 }

 #endregion
}

}

OperFactory.cs
using System;

namespace Calculate
{
 /// <summary>
 /// OperFactory 的摘要说明。
 /// 计算符接口工厂
 /// </summary>
 public class OperFactory
 {
  public OperFactory()
  {
  }
  public IOper CreateOper(string Oper)
  {
   if(Oper.Equals("+"))
   {
    IOper p = new OperAdd();
    return p;
   }
   if(Oper.Equals("-"))
   {
    IOper p = new OperDec();
    return p;
   }
   if(Oper.Equals("*"))
   {
    IOper p = new OperRide();
    return p;
   }
   if(Oper.Equals("/"))
   {
    IOper p = new OperDiv();
    return p;
   }
   return null;
  }
 }
}

Calculate.cs
using System;
using System.Collections;

namespace Calculate
{
 /// <summary>
 /// Calculate 的摘要说明。
 /// 计算实现主类
 /// </summary>
 public class Calculate
 {
  /// <summary>
  /// 算术符栈
  /// </summary>
  private ArrayList HList;
  /// <summary>
  /// 数值栈
  /// </summary>
  public ArrayList Vlist;
  /// <summary>
  /// 读算试工具
  /// </summary>
  private CalUtility cu;
  /// <summary>
  /// 运算操作器工厂
  /// </summary>
  private OperFactory of;
  /// <summary>
  /// 构造方法
  /// </summary>
  /// <param name="str">算式</param>
  public Calculate(string str)
  {
   //
   // TODO: 在此处添加构造函数逻辑
   //
   HList = new ArrayList();
   Vlist = new ArrayList();
   of = new OperFactory();
   cu = new CalUtility(str);
  }

  /// <summary>
  /// 开始计算
  /// </summary>

  public object DoCal()
  {
   string strTmp=cu.getItem();
   while(true)
   {
    if(cu.IsNum(strTmp))
    {
     //如果是数值,则写入数据栈
     Vlist.Add(strTmp);
    }
    else
    {
     //数值
     Cal(strTmp);
    }
    if(strTmp.Equals(""))
     break;
    strTmp=cu.getItem();
   }
   return Vlist[0];
  }

  /// <summary>
  /// 计算
  /// </summary>
  /// <param name="str">计算符</param>

  private void Cal(string str)
  {
   //符号表为空,而且当前符号为"",则认为已经计算完毕
   if(str.Equals("")&&HList.Count==0)
    return;
   if(HList.Count>0)
   {
    //符号是否可以对消?
    if(HList[HList.Count-1].ToString().Equals("(") && str.Equals(")"))
    {
     HList.RemoveAt(HList.Count-1);
     if(HList.Count>0)
     {
      str=HList[HList.Count-1].ToString();
      HList.RemoveAt(HList.Count-1);
      Cal(str);
     }
     return;
    }
    //比较优先级
    if(cu.Compare(HList[HList.Count-1].ToString(),str))
    {
     //如果优先,则计算
     IOper p = of.CreateOper(HList[HList.Count -1].ToString());
     if(p!=null)
     {
      Vlist[Vlist.Count -2] = p.Oper(Vlist[Vlist.Count-2],Vlist[Vlist.Count-1]);
      HList.RemoveAt(HList.Count -1);
      Vlist.RemoveAt(Vlist.Count -1);
      Cal(str);
     }
     return;
    }
    if(!str.Equals(""))
     HList.Add(str);
   }
   else
   {
    if(!str.Equals(""))
     HList.Add(str);
   }
  }
 }
}

  后记

  使用这种方法,我们可以完成更复杂的混合运算,比如说:可以支持函数运算的混合运算,更进一步,可以支持自定义函数的混合运算,最终可以将这种算法运用在自定义报表、工资处理等等应用领域,很希望有相关兴趣的朋友与我交流。

 

 

 

上下文章:

 

上一篇文章: C#中利用mediaplayer打造mp3播放器 下一篇文章: C#中利用正则表达式实现字符串搜索

相关文章:

  • 深入了解Oracle10g中新的多重集运算符
  • 在Oracle中实现数据库的复制
  • 在T-SQL中实现Oracle的MINUS集合运算符
  • Linux中实现DDOS攻击的方法
  • 如何实现纵深化的网络安全防御?

 

 

快速导航

  • 网络学院
  • 精品汇聚
  • 字体下载
  • 教程下载
  • ASP源码
  • PHP源码
  • Net源码
  • JSP 源码

编程技术分类导航

  • ASP & ASP.NET教程
  • PHP教程
  • JSP教程
  • C/C++教程
  • VB & VB.NET教程
  • VC教程
  • Delphi教程
  • BCB教程
  • VFP教程
  • PB教程
  • JAVA教程
  • XML教程
  • C#教程
  • CGI教程

本类经典文章推荐

  • Office Word 2003 精简绿色版
  • 杀恶意代码强烈推荐: 360安全卫士
  • 网络看电视首选: UUSee网络电视
  • 强烈推荐: PPS网络电视(PPStream)
  • ADO.NET的开发场景及传统ADO的处理
  • 利用Visual C#实现Window管道技术
  • C#取得汉字的拼音的首字母
  • 使用C#编写DES加密程序的framework
  • Visual C#编写3D游戏框架示例
  • 用C#和本地Windows API操纵系统菜...
  • 在C#程序设计中使用Win32类库
  • Visual C#中调用Windows服务初探
  • 如何在C#的WinForm中制作饼状图和...
  • C#中实现DataGrid双向排序

C#教程阅读排行

  • 如何在C#的WinForm中制作饼状图和...
  • 浅析C#中图形编程
  • 用C#和本地Windows API操纵系统菜...
  • 彻底剖析C# 2.0泛型类的创建和使用
  • Visual C#编写3D游戏框架示例
  • Visual C# 2005实现控件中捕获按键
  • Visual C#中调用Windows服务初探
  • Visual C#弹出窗口杀手
  • C#+Direct3D9.0开发实例之月亮绕着...
  • C#进阶教程(三)C#的数组(Arrays)

编程技术阅读总排行

  • 第二章 PowerBuilder 入门之创建新...
  • VB入门教程之一
  • 第一章 什么是PowerBuilder
  • Java连接数据库实例
  • VC++之List Box/Check List Box控...
  • VC++ List Ctrl控件
  • VC++ Combo Box/Combo Box Ex控件
  • 学C++不得不看的一篇文章
  • VB入门教程之二
  • VC++之Button控件

广告位置

关于我们 | 版权声明 | 免责条款 | 广告联系 | 软件发布 | 下载帮助 | 下载排行 | 网站地图 | 特别鸣谢 | 友情连接

copyright; 2005-2007 D9soft.com 第九软件网 版权所有