C#实现eval 进行四则运算(有码)

 

在JavaScript中实现四则运算很简单,只需要调用eval函数就行了,但是不知道什么原因万能的.NET却没有封装这个函数~

在这里为大家封装了一个C#版本的eval函数,具体的设计参考了《大话数据结构》

1. 中缀表达式

中缀表达式即我们平时使用的四则运算表达式,如:9+(3-1)*3+10/2,但是程序却很难识别这样的表达式,所以需要把它转化成后缀表达式

2. 后缀表达式

因为所有的运算符都在数字后面,所以称其为后缀表达式,如:9 3 1 – 3 * + 10 2 / +,那么程序如何识别后缀表达式呢,这里就用到了栈(Stack),主要分以下步骤:

1) 将字符从前往后进行入栈操作

2)如果字符为数字则直接入栈,如:上例中的9、3、1都直接入栈

3)如果为运算符则获取栈顶的两个数字,即执行两次pop操作,如:在执行到上例中的“-”时,将3、1出栈,然后运行3-1

4)将上一步运算的结果入栈,即将3-1=2入栈

5) 根据以上原则,运行的顺序如下:3-1=2,2*3=6,9+6=15,10/2=5,15+5=20,20就是我们想要的结果

3. 中缀表达式转后缀表达式

这里还是得用到栈(Stack),我们还是用 9+(3-1)*3+10/2 来做例子,原则如下:

  1)将字符从前往后进行入栈操作

  2)当字符为数字时直接输出,如:上例中9直接输出

  3)当字符为运算符时,如果是空栈则直接入栈,如下图步骤1

  4)如果是“(”则直接入栈,如步骤2

  5) 如果是“)”则循环取出栈内元素,直到“(”出栈,再将“()”内的运算符输出,如步骤4

  6)如果是“*”或“/”则直接入栈,如步骤5

  7)如果是“+”或“-”,且栈顶的运算符为“*”或“/”,则取出所有栈内元素输出,然后将本次操作符入栈,如步骤6

  8)表达式遍历完成后,循环取出栈内元素进行输出,我们最后得到的结果就是 9 3 1 – 3 * + 10 2 / +

4.实现源码如下:

  

 1     public class Calculator
2 {
3 public Calculator()
4 {
5 _OptStack = new Stack<char>();
6 _SuffixStack = new Stack<float>();
7 }
8
9 private Stack<char> _OptStack;
10 private Stack<float> _SuffixStack;
11
12 public float Calculate(string expression)
13 {
14 string lastNum = string.Empty;
15 for (int i = 0; i < expression.Length; i++)
16 {
17 if (char.IsNumber(expression[i]) || expression[i].Equals('.'))
18 {
19 lastNum += expression[i];
20 }
21 else
22 {
23 if (lastNum != string.Empty)
24 {
25 Merger(float.Parse(lastNum));
26 lastNum = string.Empty;
27 }
28 AddOpt(expression[i]);
29 }
30 }
31 if (lastNum != string.Empty)
32 {
33 Merger(float.Parse(lastNum));
34 }
35 while (_OptStack.Count > 0)
36 {
37 Merger(_OptStack.Pop());
38 }
39
40 return _SuffixStack.Pop();
41 }
42
43 private void AddOpt(char opt)
44 {
45 if (_OptStack.Count == 0)
46 {
47 _OptStack.Push(opt);
48 return;
49 }
50 if (opt.Equals(')'))
51 {
52 while (!_OptStack.Peek().Equals('('))
53 {
54 Merger(_OptStack.Pop());
55 }
56 _OptStack.Pop();
57 return;
58 }
59 char tempOpt = _OptStack.Peek();
60 if ((opt.Equals('-') || opt.Equals('+')) &&
61 (tempOpt.Equals('*') || tempOpt.Equals('/')))
62 {
63 while (_OptStack.Count > 0)
64 {
65 Merger(_OptStack.Pop());
66 }
67 }
68
69 _OptStack.Push(opt);
70 }
71
72 private void Merger(float exp)
73 {
74 _SuffixStack.Push(exp);
75 }
76
77 private void Merger(char exp)
78 {
79 float num1 = _SuffixStack.Pop();
80 float num2 = _SuffixStack.Pop();
81 float result = 0;
82 switch (exp)
83 {
84 case '+':
85 result = num2 + num1;
86 break;
87 case '-':
88 result = num2 - num1;
89 break;
90 case '*':
91 result = num2 * num1;
92 break;
93 case '/':
94 result = num2 / num1;
95 break;
96 }
97 _SuffixStack.Push(result);
98 }
99 }

5. 下载

Demo.rar

C#实现eval的更多相关文章

  1. JSON.parse 与 eval() 对于解析json的问题

    1.eval()与JSOn.parse的不同 eval() var c = 1; //全局变量 var jsonstr1 = '{"name":"a",&quo ...

  2. JSON.parse与eval的区别

    JSON.parse与eval和能将一个字符串解析成一个JSON对象,但还是有挺大区别. 测试代码 var A = "{ a: 1 , b : 'hello' }"; var B ...

  3. JS eval()函数的一些见解

    一.eval是基本使用规则 1 eval() 函数可计算某个字符串,并执行其中的的 JavaScript 代码. 2 eval(string) 3 string必需.要计算的字符串,其中含有要计算的 ...

  4. Javascript动态执行JS(new Function与eval比较)

    new Function与eval可以动态执行JS,只要把拼接好的JS方法,然后以字符串的形式传入到这两个函数,可以执行,其中new Function用在模板引擎比较多. 用 Function 类直接 ...

  5. Python:eval的妙用和滥用

    时间 2014-07-08 13:05:24 CSDN博客 原文  http://blog.csdn.net/zhanh1218/article/details/37562167 主题 Python ...

  6. JSON中eval与parse的区别

    json的的解析方法 (非原创) json的解析方法共有两种:eval_r() 和 JSON.parse(),使用方法如下: var jsonData = '{"data1":&q ...

  7. eval解析JSON中的注意点

       在JS中将JSON的字符串解析成JSON数据格式,一般有两种方式: 1.一种为使用eval()函数. 2. 使用Function对象来进行返回解析. 使用eval函数来解析,并且使用jquery ...

  8. jQuery Ajax请求(关于火狐下SyntaxError: missing ] after element list ajax返回json,var json = eval("("+data+")"); 报错)

    $.ajax({    contentType: "application/x-www-form-urlencoded;charset=UTF-8" ,    type: &quo ...

  9. 被嫌弃的eval和with

    × 目录 [1]eval [2]with 前面的话 eval和with经常被嫌弃,好像它们的存在就是错误.在CSS中,表格被嫌弃,在网页中只是用表格来展示数据,而不是做布局,都可能被斥为不规范,矫枉过 ...

  10. Global eval. What are the options?

    David Flanagan最近写了一个关于全局eval的简单表达式,可以用一行式子表示: var geval = this.execScript || eval; 尽管看起来很简短,但是跨浏览器的兼 ...

随机推荐

  1. 左右lcm,gcd一些性质

    两个整数a,b  他们的最大公约数为n  最小公倍数为m  则有 a,b都能分解为有限个素数的积               12 = 2^2 * 3^1 * 5^0 , 30 = 2^1 * 3^1 ...

  2. JavaScript两种方法来定义一个函数

    <!DOCTYPE html> <html> <head> <meta charset="UTF-8"> <title> ...

  3. unity3d插件Daikon Forge GUI 中文教程-1-Daikon Forge介绍

    (游戏蛮牛首发)大家好我是孙广东官网提供了专业的视频教程http://www.daikonforge.com/dfgui/tutorials/,只是是在youtube上,要观看是须要FQ的. 只是教程 ...

  4. 【phpMyAdmin】更改配置文件连接到其他server

    默认phpMyAdmin安装完毕后对机器的访问mysql,但有时我们需要访问其它server的mysql数据库,所以我们需要配置. 真,phpMyAdmin已经为我们做了配置的选项.可是须要我们进行一 ...

  5. .NET到Node.js

    从.NET到Node.js谈前后端分离实践(by vczero)   一.最初的[无分离]实践 11年末的时候,用winForm开发程序,拖拖控件,点点按钮,连接数据库,做一些基本的管理系统:Java ...

  6. Wowza流媒体Live直播和VOD点播配置实战-attach

    Wowza是当今可以说最流行的流媒体服务器之一,近来因为需要搭建相应的服务器,但又不想用camera等作真实的直播,所以想办法用媒体文件转换成直播流再提供给Wowza进行直播.这里把该设置步骤以及设计 ...

  7. APP-随身听

    简单到复杂听你的专属音响界,听金融.听物业,听新闻和其他节目专辑,简要介绍了新的音频应用,给你不一样的聆听体验.还记得老歌做?这里有.您留声机的一部分!很简单的音频应用,随时随地与此应用程序来听你的私 ...

  8. leetcode第二题--Median of Two Sorted Arrays

    Problem:There are two sorted arrays A and B of size m and n respectively. Find the median of the two ...

  9. Android-异步图像装载机

    在ListView加载图像是非常常见的场景,图像加载几个要求满足以下的: (1)是否画面位于网络或本地上,装载不应同步.但应该异步加载,例如,使用AsyncTask. (2)为了避免重复下载图片和网页 ...

  10. Codeforces 10D LCIS 找出最长公共子和产量增加这个序列 dp

    主题链接:点击打开链接 意甲冠军: 特定n长序列 给定k长序列 求LCIS并输出这个子序列 如有多解输出随意解.. = - = 敲的时候听着小曲儿pre的含义还没有想清楚,万万没想到就过了... #i ...