题目链接

描述

聪明的你帮助C小加解决了中缀表达式到后缀表达式的转换(详情请参考“郁闷的C小加(一)”),C小加很高兴。但C小加是个爱思考的人,他又想通过这种方法计算一个表达式的值。即先把表达式转换为前缀和后缀表达式,再求值。这时又要考虑操作数是小数和多位数的情况。

  • 输入

    第一行输入一个整数T,共有T组测试数据(T<10)。每组测试数据只有一行,是一个长度不超过1000的字符串,表示这个运算式,每个运算式都是以“=”结束。这个表达式里只包含+-*/与小括号这几种符号。其中小括号可以嵌套使用。数据保证输入的操作数中不会出现负数并且小于1000000。数据保证除数不会为0。
  • 输出

    对于每组测试数据输出结果包括三行,先输出转换后的前缀和后缀表达式,再输出计算结果,结果保留两位小数。
  • 样例输入

    2

    1+2=

    (19+21)*3-4/5=
  • 样例输出

    + 1 2 =

    1 2 + =

    3.00

    - * + 19 21 3 / 4 5 =

    19 21 + 3 * 4 5 / - =

    119.20

分析:

我天,二百多行的代码·····

之前一直做的表达式求值都在中缀转换成后缀,然后直接进行计算的,但是这道题还要求输出表达式的前缀形式。本来对于中缀转前缀形式就只是简单的理解,并不太懂要怎么转换,所以很头痛,仔细了解过之后才知道前缀与后缀在转换过程中的原理是一样的,只是后缀是从前到后转换,再从前到后输出来;而前缀是从后到前转换,淡然最后也要从后到前输出来,其他的都没有什么太大的区别。

代码:

#include<string>
#include<iostream>
#include<algorithm>
#include<stdio.h>
#include<string.h>
#include<stack>
using namespace std;
char a[1009],b[1009],c[1009];
char pri[300];
char pri1[300];
//
void trans(char a[])///中缀转后缀
{
stack<char>q;///栈用来存储操作数
q.push('=');///用'='压栈,是为了与后来的操作符比较的时候,他的优先级最低
int j=0;
int k=0;
for(int i=0; a[i]!='='; i++)
{
if(a[i]>='0'&&a[i]<='9'||a[i]=='.')///可能有小数
{
b[j++]=a[i];
k++;
} else
{
if(k!=0)
{
b[j++]=' ';
k=0;
}
if(a[i]=='(')
q.push(a[i]);
else if(a[i]==')')///一直出栈到'('
{
while(q.top()!='(')
{
b[j++]=q.top();
q.pop();
b[j++]=' ';
}
q.pop();
}
else
{
while(pri[a[i]]<=pri[q.top()])///将所有的运算符优先级比当前小的出来
{
b[j++]=q.top();
q.pop();
b[j++]=' ';
}
q.push(a[i]);///在将当前的运算符进栈
}
}
}
int nn=j-1;
if(b[nn]!=' ')
b[j++]=' ';
while(!q.empty()&&q.top()!='=')///栈中的元素也要全部输出
{
b[j++]=q.top();
q.pop();
b[j++]=' ';
}
b[j++]='=';
b[j]='\0';
puts(b);
} void Trans(char a[])///中缀转前缀
{
stack<char>st;
int len=strlen(a);
int j=0;
int k=0;
for(int i=len-1; i>=0; i--)///逆着转换
{
if(a[i]>='0'&&a[i]<='9'||a[i]=='.')
{
c[j++]=a[i];
k++;
}
else
{
if(k!=0)
{
c[j++]=' ';
k=0;
}
if(a[i]==')')///与转后缀相反,')'直接进栈
{
st.push(a[i]);
}
else if(a[i]=='(')///'('的话要一直找到')'
{
while(st.top()!=')')
{
c[j++]=st.top();
st.pop();
c[j++]=' ';
}
st.pop();
}
else
{
while(!st.empty()&&pri1[st.top()]>pri1[a[i]])///将栈中的优先级比当前大的出栈
{
if(st.top()!='=')
{
c[j++]=st.top();
st.pop();
c[j++]=' ';
}
}
st.push(a[i]);
}
}
}
int hh=j-1;
if(c[hh]!=' ')
c[j++]=' ';
while(!st.empty()&&st.top()!='=')
{
c[j++]=st.top();
c[j++]=' ';
st.pop();
}
int nn=j-1;
if(c[nn]==' ')
nn=j-2;
else
nn=j-1;
for(int kk=nn;kk>=0;kk--)///最后也是要逆着输出来的
printf("%c",c[kk]);
printf(" =\n");
}
double JiSuan(char b[])
{
//puts(b);
char c[100];
memset(c,'\0',sizeof(c));
int j=0;
stack<double>st;
for(int i=0; b[i]!='='; i++)
{
if(b[i]>='0'&&b[i]<='9'||b[i]=='.')
c[j++]=b[i];
else
{ if(j!=0)
{
st.push(atof(c));///自动将字符转换为double
// cout<<"c "<<st.top()<<endl;
memset(c,'\0',sizeof(c));
j=0;
}
if(b[i]!=' ')
{
double n1;
double n2;
double n3;
switch(b[i])
{
case '+':
n1=st.top();
st.pop();
n2=st.top();
st.pop();
n3=n2+n1;
st.push(n3);
break;
case '-':
n1=st.top();
st.pop();
n2=st.top();
st.pop();
n3=n2-n1;
st.push(n3);
break;
case '*':
n1=st.top();
st.pop();
n2=st.top();
st.pop();
n3=n2*n1;
st.push(n3);
break;
case '/':
n1=st.top();
st.pop();
n2=st.top();
st.pop();
n3=n2/n1;
st.push(n3);
break;
}
}
} }
return st.top();
}
int main()
{
int T;
pri['=']=-1;
pri['(']=0;
pri['+']=1;
pri['-']=1;
pri['*']=2;
pri['/']=2;
pri[')']=3;
pri1['=']=-1;
pri1[')']=0;
pri1['+']=1;
pri1['-']=1;
pri1['*']=2;
pri1['/']=2;
pri1['(']=3;
scanf("%d",&T);
for(int k=1; k<=T; k++)
{
memset(a,'\0',sizeof(a));
memset(b,'\0',sizeof(b));
memset(c,'\0',sizeof(c));
scanf(" %s",a);
// puts(a);
Trans(a);
trans(a);
/// puts(b);
printf("%.2lf\n",JiSuan(b));
}
return 0;
}

NYOJ 409 郁闷的C小加(三) (字符串处理)的更多相关文章

  1. nyoj 409——郁闷的C小加(三)——————【中缀式化前缀后缀并求值】

    郁闷的C小加(三) 时间限制:1000 ms  |  内存限制:65535 KB 难度:4   描述 聪明的你帮助C小加解决了中缀表达式到后缀表达式的转换(详情请参考“郁闷的C小加(一)”),C小加很 ...

  2. NYOJ 267 郁闷的C小加(二) (字符串处理)

    题目链接 描述 聪明的你帮助C小加解决了中缀表达式到后缀表达式的转换(详情请参考"郁闷的C小加(一)"),C小加很高兴.但C小加是个爱思考的人,他又想通过这种方法计算一个表达式的值 ...

  3. NYOJ 257 郁闷的C小加(一)

    郁闷的C小加(一) 时间限制:1000 ms  |  内存限制:65535 KB 难度: 描写叙述 我们熟悉的表达式如a+b.a+b*(c+d)等都属于中缀表达式.中缀表达式就是(对于双目运算符来说) ...

  4. nyoj 257 郁闷的C小加(一)(栈、队列)

    郁闷的C小加(一) 时间限制:1000 ms  |  内存限制:65535 KB 难度:3   描述 我们熟悉的表达式如a+b.a+b*(c+d)等都属于中缀表达式.中缀表达式就是(对于双目运算符来说 ...

  5. NYOJ 257 郁闷的C小加(一) (字符串处理)

    题目链接 描述 我们熟悉的表达式如a+b.a+b(c+d)等都属于中缀表达式.中缀表达式就是(对于双目运算符来说)操作符在两个操作数中间:num1 operand num2.同理,后缀表达式就是操作符 ...

  6. NYOJ-267 郁闷的C小加(二)

    郁闷的C小加(二) 时间限制:1000 ms  |  内存限制:65535 KB 难度:4   描述 聪明的你帮助C小加解决了中缀表达式到后缀表达式的转换(详情请参考“郁闷的C小加(一)”),C小加很 ...

  7. nyoj-257 郁闷的C小加(一) 前缀表达式变后缀

    郁闷的C小加(一) 时间限制:1000 ms  |  内存限制:65535 KB 难度:3   描述 我们熟悉的表达式如a+b.a+b*(c+d)等都属于中缀表达式.中缀表达式就是(对于双目运算符来说 ...

  8. 郁闷的C小加(一)(后缀表达式)

    郁闷的C小加(一) 时间限制:1000 ms  |  内存限制:65535 KB 难度:3   描述 我们熟悉的表达式如a+b.a+b*(c+d)等都属于中缀表达式.中缀表达式就是(对于双目运算符来说 ...

  9. nyoj 236 心急的C小加

    心急的C小加 时间限制:1000 ms  |  内存限制:65535 KB 难度:4   描述 C小加有一些木棒,它们的长度和质量都已经知道,需要一个机器处理这些木棒,机器开启的时候需要耗费一个单位的 ...

随机推荐

  1. Windows下多线程编程(一)

    前言 熟练掌握Windows下的多线程编程,能够让我们编写出更规范多线程代码,避免不要的异常.Windows下的多线程编程非常复杂,但是了解一些常用的特性,已经能够满足我们普通多线程对性能及其他要求. ...

  2. virtualenv是什么?virtualenv的安装及pycharm的配置和使用

    virtualenv是什么? virtualenv是一个创建隔绝的Python环境的工具.virtualenv创建一个包含所有必要的可执行文件的文件夹,用来使用Python工程所需的包.简单的说就是一 ...

  3. 'phantomjs.exe' executable needs to be in PATH. (selenium PhantomJS python)

    今天selenium PhantomJS python用了下,发现报错,提示我:'phantomjs.exe' executable needs to be in PATH. from seleniu ...

  4. 【前端学习笔记】call、apply、bind方法

    1.call()方法: // move函数实现移动平面图上一个点位置功能 var move = function(x,y){ this.x += x; this.y += y; } // 定一个点p ...

  5. c#对一个类的扩展

    首先定义一个静态类,参数使用this约束并选择需要扩展的类,当然也可以 继续添加扩展是需要添加的参数 public static class StringExrprp { /// <summar ...

  6. bzoj4032-最短不公共子串

    题意 给出两个长度小于等于2000的小写字母串,四个问题: A的最短子串不是B的子串 A的最短子串不是B的子序列 A的最短子序列不是B的子串 A的最短子序列不是B的子序列 分析 虽然求的是不公共,但是 ...

  7. BZOJ 2190 仪仗队(线性筛欧拉函数)

    简化题意可知,实际上题目求得是gcd(i,j)=1(i,j<=n)的数对数目. 线性筛出n大小的欧拉表,求和*2+1即可.需要特判1. # include <cstdio> # in ...

  8. 【bzoj4027】[HEOI2015]兔子与樱花 树形dp+贪心

    题目描述 很久很久之前,森林里住着一群兔子.有一天,兔子们突然决定要去看樱花.兔子们所在森林里的樱花树很特殊.樱花树由n个树枝分叉点组成,编号从0到n-1,这n个分叉点由n-1个树枝连接,我们可以把它 ...

  9. BZOJ5011 JXOI2017颜色(主席树)

    相当于求满足在子段中出现的颜色只在该子段中出现的非空子段数量.这也就相当于其中出现的颜色最左出现的位置在左端点右侧,最右出现的位置在右端点左侧.那么若固定某个端点,仅考虑对该端点的限制,会有一段合法区 ...

  10. BZOJ4998 星球联盟(LCT+双连通分量+并查集)

    即要求动态维护边双.出现环时将路径上的点合并即可.LCT维护.具体地,加边成环时makeroot+access+splay一套把这段路径提出来,暴力dfs修改并查集祖先,并将这部分与根断开,视为删除这 ...