计算表达式。

只有3种运算符:*,+,- ,

*优先级高于后两者,后两者优先级相同。

有两种符号:{},()。

利用递归和堆栈即可解决。

首先遇到左括号开始入栈直到遇到右括号,遇到右括号时对括号内的数进行计算。

考虑到*优先级较高,因此遇到*直接对其左右集合进行运算。

最后得到不含括号和*的表达式,从左往右计算即可。

http://poj.org/problem?id=2269

 #include <cstdio>
#include <cstring>
#include <algorithm>
const int maxn = ;
const int maxc = ;
char buf[maxn];
int stack[maxn][maxc];
char op[maxn];
int p, k;
int vis[maxc]; void caculate(){
//meeting eol or right paren RETURN
if(buf[p] == '\0' || buf[p] == ')') return;
if(buf[p] == '('){
int k1 = k;
++p, caculate();
//guarantee that all contents in the paren computed
//figure out the current result
for(int i = k1 + ; i < k; i++){
memset(vis, , sizeof vis);
int o = op[i] == '+' ? : -;
for(int j = ; j <= stack[k1][]; j++) ++vis[stack[k1][j]];
for(int j = ; j <= stack[i][]; j++) vis[stack[i][j]] += o;
stack[k1][] = ;
for(int j = ; j < ; j++) if(vis[j] > ) stack[k1][++stack[k1][]] = j;
}
k = k1 + ;
++p, caculate();
}
else if(buf[p] == '{'){
++p;
stack[k][] = ;
for(int i = p; buf[i] != '}'; i++, ++p) stack[k][++stack[k][]] = buf[i] - 'A';
k++;
p++;
caculate();
}else if(buf[p] == '*'){
if(buf[p + ] == '(') ++p, caculate();
else if(buf[p + ] == '{'){
++p;
stack[k][] = ;
for(int i = p; buf[i] != '}'; i++, ++p) stack[k][++stack[k][]] = buf[i] - 'A';
k++;
++p;
}
//caculate stack[k - 2] * stack[k - 1] to update stack[k - 2]
memset(vis, , sizeof vis);
for(int i = ; i <= stack[k - ][]; i++) ++vis[stack[k - ][i]];
for(int i = ; i <= stack[k - ][]; i++) ++vis[stack[k - ][i]];
stack[k - ][] = ;
for(int i = ; i < ; i++) if(vis[i] == ) stack[k - ][++stack[k - ][]] = i;
--k;
caculate();
}else{
op[k] = buf[p];
++p;
caculate();
}
} void solve(){
k = p = ;
memset(vis, , sizeof );
caculate();
for(int i = ; i < k; i++){
memset(vis, , sizeof vis);
int o = op[i] == '+' ? : -;
for(int j = ; j <= stack[i - ][]; j++) ++vis[stack[i - ][j]];
for(int j = ; j <= stack[i][]; j++) vis[stack[i][j]] += o;
stack[i][] = ;
for(int j = ; j < ; j++) if(vis[j] > ) stack[i][++stack[i][]] = j;
}
putchar('{');
for(int i = ; i <= stack[k - ][]; i++) putchar(stack[k - ][i] + 'A');
putchar('}');
putchar('\n');
} int main(){
#ifndef ONLINE_JUDGE
freopen("in.txt", "r", stdin);
#endif // ONLINE_Judge
while(gets(buf) && buf[] != '\0') solve();
return ;
}

poj2269 Friends的更多相关文章

随机推荐

  1. Radius session

    1,EAP 中继 client start, NAS require identity, client sent username, NAS sent username to sever, serve ...

  2. Moment.js学习(一)源代码

    本篇主要是学习Moment.js.类库源代码如下: 2.4版本. //! moment.js //! version : 2.4.0 //! authors : Tim Wood, Iskren Ch ...

  3. struts 2.0部署

    环境:linux centos 64位. 1)下载JDK6.0,具体文件名是:jdk-6u45-linux-x64.bin 安装:chmod 755 jdk*.bin ./jdk....bin 设置环 ...

  4. Lintcode: Rotate String

    Given a string and an offset, rotate string by offset. (rotate from left to right) Example Given &qu ...

  5. session 实现保存用户信息

    index.jsp <body> <div style="margin: 0 auto; width: 500px; text-align: center;"&g ...

  6. 20145207 《Java程序设计》第4周学习总结

    前言 又到了大家最喜欢的前言时间,哈哈哈.我这个人啊,就是比较爱闲聊.正式在学校呆的第一天时间就在这里敲代码,自己都觉得自己伟大.不过好无聊呀....这周的内容说实话讲我还是挺感兴趣的,因为书上的例子 ...

  7. 。。。Hibernate中mappedBy属性。。。

    今天在学习Hibernate中,感觉这个mappedBy这个注解属性有点小难度.不过理解之后,还是阔以的! 首先,mappedBy这个注解只能够用在@OntToOne,@OneToMany,@many ...

  8. paper 60 :转载关于视觉SCI期刊

    ChanLee_1整理的计算机视觉领域稍微容易中的期刊 模式识别,计算机视觉领域,期刊 (1)pattern recognition letters, 从投稿到发表,一年半时间 (2)Pattern  ...

  9. angular 倒计时

    $scope.countdown = ; var myTime = setInterval(function() { $scope.countdown--; $scope.$digest(); // ...

  10. Linux下用SCP无需输入密码传输文件

    在Linux环境下,两台主机之间传输文件一般使用scp命令,通常用scp命令通过ssh获取对方linux主机文件的时候都需要输入密码确认. 不过通过建立信任关系,可以实现不输入密码. 这里假设A的IP ...