注册 登录
编程论坛 C++教室

[原创]一个计算四则表达式的模板

雨中飞燕 发布于 2007-09-09 10:18, 6788 次点击
*/ --------------------------------------------------------------------------------------
*/ 出自: 编程中国 https://www.bc-cn.net
*/ 作者: 雨中飞燕 QQ:609754650
*/ 时间: 2007-9-9 编程论坛首发
*/ 声明: 尊重作者劳动,转载请保留本段文字
*/ --------------------------------------------------------------------------------------

在9月8日那天我特意编写的,给大家分享的,
一个很方便的函数模板,可以并且只可以计算含括号的四则表达式
只有一个函数接口:int GetExpValue(_Tstream& istrin, _T& nReturn)
参数解释:
istrin: 一个输入流,可以是标准IO流,可以是文件流,也可以是串流
nReturn:用于接收计算结果的变量,计算所使用的类型由这个变量确定
返回值:
返回非0表示计算成功,0表示计算失败有错误

程序代码:
namespace fy_Exp{
namespace {template <class _T>
inline _T GetExpValue(_T t[], char& csym){
char c=csym; csym=0;
switch(c){
case '+':return t[0] += t[1];
case '-':return t[0] -= t[1];
case '*':return t[0] *= t[1];
default: return t[0] /= t[1];//case '/':
}
}}
template <class _T, class _Tstream>
/* istrin: inputstream, nReturn: get return value
Return nonzero if get value successfully */
int GetExpValue(_Tstream& istrin, _T& nReturn){
_T t[3] = {0}; //雨中飞燕之作
char csym[3] = "++";
int nLevel = 1, nERR = 0;
if(!(istrin>>t[1]))istrin.clear();
for(;;){
if(istrin>>csym[2]){
switch(csym[2]){
case '(':
if(!csym[1]){nLevel=0x100; nERR=1;}else
if(!GetExpValue(istrin, t[2]))nLevel|=0x10;
else{nLevel=0x100; nERR=1;}
break;
case ')':
{nLevel = 0x100;}break;
case '+':case '-':case '*':case '/':
{csym[nLevel++] = csym[2];}break;
case ' ':case '\r':case '\n':case '\t':continue;
default:
{nLevel=0x100; nERR=1;}
}
if(nLevel==0x100)break;
if(nLevel&0x10 || istrin>>t[2]){
nLevel &= 0xF;
if(nLevel==1){t[1]=t[2];csym[1]=0;continue;}
if(csym[1]=='*'||csym[1]=='/'){
GetExpValue(t+1, csym[1]);
}
else{
GetExpValue(t, csym[0]);
t[1]=t[2];csym[0]=csym[1];csym[1]=0;
}
nLevel = 1;
}
else istrin.clear();
}
else{nERR = -1; break;}
}
if(csym[1])t[2]=0,nReturn=GetExpValue(t+1, csym[1]);
else nReturn=GetExpValue(t, csym[0]);
return nERR==-1?1:0;
}}



函数模板使用示例:
在以上那段代码的后面加上以下代码:

程序代码:
#include<strstream>
#include<iostream>
#include<string>
using namespace std;
int main(void)
{
string s1;
while(cin>>s1)
{
istrstream isin(s1.data());
double d;
if(fy_Exp::GetExpValue(isin, d))
{
cout<<d<<endl;
}
else
{
cout<<"ERROR"<<endl;
}
}
return 0;
}


然后编译执行就可以了(*^_^*)
其它:TC++上一定编译错误,不保证在VC6上也能通过编译
建议使用VC7或VC更高版本,或者使用GNU C++编译

如果对以上代码有疑问或者建议都可以找我。
21 回复
#2
雨中飞燕2007-09-09 10:37
刚刚再修改了一下,现在应该都正确了
要是哪个表达式计算不正确请告诉一声。。。。thanks
#3
野比2007-09-09 10:53
终于可以上网了.. 先收下,有空慢慢看..
#4
aipb20072007-09-09 11:12
空了再看看哈!
#5
zhangwei2007-09-09 14:05
程序一句注释也没有?
#6
雨中飞燕2007-09-09 14:09
除了接口解释那个注释外,其它的我去掉了
#7
冰的热度2007-09-09 14:10
什么东东,这么好,居然加了精,我来看看......
#8
chen78062007-09-13 08:24

我用dev编译没有通过,本来想试试类模板,可是没机会.

#9
雨中飞燕2007-09-13 10:25
以下是引用chen7806在2007-9-13 8:24:43的发言:

我用dev编译没有通过,本来想试试类模板,可是没机会.


我试过用G++编译,是通过的,
我也肯定我的代码在G++上一定能够通过,至于你自己是怎么弄的不通过我就不明白了

#10
yuki2007-09-16 09:37

我想说说我个人的看法,若果楼主认为不可取,可以忽略:

1、一个很明显的bug,试试看输入1 + 2 (中间是空白,会造成什么样的结果)
我这里的运行结果是:
1
0
2
2、整个算法使用递归来计算,虽然代码空间节约了,可是丧失的是运行效率。
3、还有一个问题,你可以试试看1++1这样一个表达式,照理说应该给出错误表达式的提示,而为什么能照样计算?
4、直接输入运算符而不输入数值的话也可以运算,但会造成意想不到的结果。
5、再循环中如果分支语句嵌套太深的话,非常影响效率,对于输入字符串流不怎么长的情况下,并无大碍。
6、虽然本算法对符号处理还是有点欠缺的地方,不过对于负数和实数的处理还是比较不错的。

给个建议:
现在热爱和能够坚持编程的女孩子不多,论坛偶尔能出一两个你这样对程序执著的女孩也不错;不过,编程好比练功,你招式再多,没有很好的内功是成不了高手的,建议在学习语言的同时,将《数据结构》、《操作系统》等诸类计算课程的功底打扎实,那样的话以后盖高楼就不累了;对于自己稍微有点成就,请不要骄傲与自满,多听取别人的意见,特别是批评的话,对自己的成长绝对有好处,要知道一个编程的人并不是只有技术就行了,更多的要有素养与内涵;最后,请多写程序,从实践中体会理论,去总结理论,不用对于一些复杂的算法的数学模型感到望而却步,往往实践过后,先前感到理论的困难便会烟消云散,实践是检验真理的唯一途径;最后祝你学业有成!

[此贴子已经被作者于2007-9-16 9:45:25编辑过]

#11
雨中飞燕2007-09-16 09:46
1中间有空白出现这个问题是因为cin本身,这个你可以自己去修改输入方式,这个不是bug
你自己可以手动改成getline
2效率是没有降低多少的,字符只扫描一次就完成计算
3你说的第三个问题在我新版本代码中没有此问题,但我不会帖出那个
4同上
5同2,嵌套深不深不是算法效率的关键
#12
雨中飞燕2007-09-16 09:48
从你的提问我知道的是你或许算法强,不过不知道你为什么把cin遇到空格截断这个问题当成我的程序bug
#13
yuki2007-09-16 10:01

首先声明我在这里只是谈谈我的想法,并没有和任何人比程序强不强,算法厉不厉害的问题。

你交付我的程序,让我测试,那么我现在就是扮演一个测试员的身份,程序员当然把编译好的代码给测试员进行测试,哪有让测试员改代码的呢?

最为一个测试员,我只理解你程序的使用方法与运行结果,对于内在的机理我不感兴趣,当然为了软件的品质,各种输入情况都得验证不是吗?

为什么将表达式之间的空格作为bug,理所应当罗,粗心的用户在输入表达试时不小心引入了空格,结果运行结果不是预期的,难道用户会帮程序员说话,你的程序没有问题,使我操作问题吗?当然测试员就是扮演这样一个特殊的用户角色,对于遇到问题难道要自己修改吗?那么要开发人员干嘛呢?

#14
雨中飞燕2007-09-16 10:05
算了,不和你争辩这个没有意义的问题
你要是有兴趣就把你自己的代码发到http://上测试一下吧
#15
雨中飞燕2007-09-16 10:38
还有一个问题:
1++1这样一个表达式,可以解释为1加上“正1”
表达式合法,谢谢
如果你说这个表达式错,那我就不知道能说什么了
#16
aipb20072007-09-16 12:57

支持yuki!

#17
HJin2007-09-16 15:02
good job, sister.

good comments, brother.
#18
once_again2007-10-16 17:21
程序的风格好像不太好,这程序吗,适用于研究的吗。所以应该写清楚一点,看的好累,不过还是要对楼主无私的奉献表示感谢。
感谢
#19
zwbs007zl0072007-10-17 13:32
看不懂啊
#20
卷入内核2007-10-17 20:59
大家交流下
本人也写过一个四则运算式,同意楼上的兄弟说法,我写的程序早几个月就在这上面发了,不过是用C写的,有空去看看,有报错机制,不过有等完善....
呵,有空大家帮看下

https://bbs.bc-cn.net/viewthread.php?tid=158243&star=at#

[此贴子已经被作者于2007-10-17 21:05:34编辑过]

#21
StarWing832008-04-15 20:36
我都准备写一个了……恩……
那个空格问题,可以先get(),然后看情况unget(),再operator>>解决的……
#22
sunkaidong2008-04-15 20:54
我刚帮别人写了。。用c,挺好玩得。。
1