ZOJ 1958. Friends
题目链接:
题目简介:
(1)题目中的集合由 A-Z 的大写字母组成,例如 "{ABC}" 的字符串表示 A,B,C 组成的集合。
(2)用运算符三种集合运算,'+' 表示两个集合的并集,'*' 表示两个集合的交集, '-' 表示从第一个集合中排除第二个集合包含的元素。
(3)给出这样的表达式,求出表达式结果(按照字母顺序)。运算符优先级和编程语言中的规定相同,即优先级从高到低为括号,乘号,加/减号;相同优先级时从左向右。
例如: "{ABC}+{BCD}" = "{ABCD}";
题目分析:
属于常规的表达式求值,方法是借助两个栈来解析。本质上来讲,这是一个简单题目,但是细节比较繁琐,要写对需要耐心。集合可以用 bitset 来表示这个集合,由于集合的元素数目不会超过 26,因此用 uint32 整数即可表示集合,这样可以把集合运算转换为整数的位操作来完成,比直接处理字符串更加方便。例如:
s1: { A } -> n1 = 0x01;
s2: { B } -> n2 = 0x02;
则:(s1+s2)转换为(n1 | n2)。
题目声明输入数据都是合法的表达式,不需要考虑表达式合法性判断。则只需要表达式的解析,解析过程由运算符驱动,其中需要注意的细节有:
(1)左括号在栈内和栈外的优先级需要区分对待(在栈外具有最高优先级,入栈后则优先级降低,以让后续的运算符能够入栈)。
(2)右括号实际上不入栈,而是作为一个信号,负责把和它匹配的左括号从栈中弹出。
(3)在解析前,提前在运算符栈中压入一个左括号,这样编码形式就会统一(不需要考虑在最开始阶段,取不到栈内的运算符优先级的问题)。
(4)解析结束后,取出操作数栈中的唯一元素即可。再将其解析为字符串形式。
需要考虑的特殊情况比如:
({A})+({B}) 相当于:(1)+(2);
提交代码:
#include <stdio.h>
#include <string.h> char stack_op[];
unsigned int stack_num[];
int top_op;
int top_num; int GetPrecedence(char op, int bInStack);
char* GetNumber(char *str, unsigned int *pNum);
unsigned int Compute(char* line);
int main(int argc, char* argv[]); int GetPrecedence(char op, int bInStack)
{
switch(op)
{
case : return ;
case '(': return bInStack? : ;
case ')': return ;
case '+': return ;
case '-': return ;
case '*': return ;
}
return ;
} /*
{ABCDEFG}+
| |
str ret_val
*/
char* GetNumber(char *str, unsigned int *pNum)
{
char *pCh = str + ;
*pNum = ; while(*pCh != '}')
{
if(*pCh >= 'A' && *pCh <= 'Z')
*pNum = *pNum | ( << (*pCh - 'A')); ++pCh;
}
return pCh;
} unsigned int Compute(char* line)
{
char *pCh;
unsigned int n1, n2, result;
int prec1, prec2; top_op = ;
stack_op[top_op] = '('; top_num = -;
pCh = line; while()
{
if(*pCh == '{')
{
pCh = GetNumber(pCh, &n1);
++top_num;
stack_num[top_num] = n1;
}
else
{
prec2 = GetPrecedence(*pCh, );
prec1 = GetPrecedence(stack_op[top_op], );
if(prec2 > prec1)
{
++top_op;
stack_op[top_op] = *pCh;
}
else
{
while(prec2 <= prec1 && strchr("*+-", stack_op[top_op]) != NULL)
{
n1 = stack_num[top_num - ];
n2 = stack_num[top_num];
switch(stack_op[top_op])
{
case '+': result = (n1 | n2); break;
case '-': result = (n1 & (~n2)); break;
case '*': result = (n1 & n2); break;
}
--top_num;
stack_num[top_num] = result;
--top_op;
prec1 = GetPrecedence(stack_op[top_op], );
} if(*pCh == ')')
{
while(stack_op[top_op] != '(')
{
--top_op;
}
--top_op;
}
else if(*pCh == )
break;
else
{
/* push current operator into stack */
++top_op;
stack_op[top_op] = *pCh;
}
}
}
++pCh;
} /* __ENDOF__ while(1) */ if(top_num == )
result = stack_num[];
else
result = ;
return result;
} int main(int argc, char* argv[])
{
unsigned int result, x;
char line[]; while(gets(line) != NULL)
{
result = Compute(line);
printf("{");
for(x = ; x < ; x++)
{
if(result & ( << x))
printf("%c", x + 'A');
}
printf("}\n");
}
return ;
}
zoj_1958_code
提交结果:
| Judge Status | Problem ID | Language | Run Time(ms) | Run Memory(KB) |
| Accepted | 1958 | C | 0 | 172 |
ZOJ 1958. Friends的更多相关文章
- ZOJ题目分类
ZOJ题目分类初学者题: 1001 1037 1048 1049 1051 1067 1115 1151 1201 1205 1216 1240 1241 1242 1251 1292 1331 13 ...
- ZOJ People Counting
第十三届浙江省大学生程序设计竞赛 I 题, 一道模拟题. ZOJ 3944http://www.icpc.moe/onlinejudge/showProblem.do?problemCode=394 ...
- ZOJ 3686 A Simple Tree Problem
A Simple Tree Problem Time Limit: 3 Seconds Memory Limit: 65536 KB Given a rooted tree, each no ...
- ZOJ Problem Set - 1394 Polar Explorer
这道题目还是简单的,但是自己WA了好几次,总结下: 1.对输入的总结,加上上次ZOJ Problem Set - 1334 Basically Speaking ac代码及总结这道题目的总结 题目要求 ...
- ZOJ Problem Set - 1392 The Hardest Problem Ever
放了一个长长的暑假,可能是这辈子最后一个这么长的暑假了吧,呵呵...今天来实验室了,先找了zoj上面简单的题目练练手直接贴代码了,不解释,就是一道简单的密文转换问题: #include <std ...
- ZOJ Problem Set - 1049 I Think I Need a Houseboat
这道题目说白了是一道平面几何的数学问题,重在理解题目的意思: 题目说,弗雷德想买地盖房养老,但是土地每年会被密西西比河淹掉一部分,而且经调查是以半圆形的方式淹没的,每年淹没50平方英里,以初始水岸线为 ...
- ZOJ Problem Set - 1006 Do the Untwist
今天在ZOJ上做了道很简单的题目是关于加密解密问题的,此题的关键点就在于求余的逆运算: 比如假设都是正整数 A=(B-C)%D 则 B - C = D*n + A 其中 A < D 移项 B = ...
- ZOJ Problem Set - 1001 A + B Problem
ZOJ ACM题集,编译环境VC6.0 #include <stdio.h> int main() { int a,b; while(scanf("%d%d",& ...
- zoj 1788 Quad Trees
zoj 1788 先输入初始化MAP ,然后要根据MAP 建立一个四分树,自下而上建立,先建立完整的一棵树,然后根据四个相邻的格 值相同则进行合并,(这又是递归的伟大),逐次向上递归 四分树建立完后, ...
随机推荐
- dede 优化打开速度
织梦DedeCMS本地后台运行速度慢 不知道从什么时候开始,织梦DedeCMS在本地PHP环境进行测试的时候,后台的运行反应会非常的慢,经常过了很久都没有反应.运行很久之后,还会出现了“\includ ...
- 通过gradle来导入jar包
1.通过gradle配置第三方jar包 我们看到,每个module都有一个build.gradle文件,它其实是对应module的配置文件.关于build.gradle文件中具体内容的含义,我们将在最 ...
- mysql 完整性约束
mysql 完整性约束 数据的完整性概述根据完整性实施的方法将完整性约束分为四类:1.实体完整性 实体完整性的实现:通过在表中设置主键约束.唯一约束或标识列来实现 主键约束:应用于表列的一个约束 用法 ...
- linux计划任务
一.单一计划任务 安装at: # yum -y install at 启动: # /etc/init.d/atd start 查看是否运行: # ps aux | grep atd 创建计划 # at ...
- Digital root(数根)
关于digital root可以参考维基百科,这里给出基本定义和性质. 一.定义 数字根(Digital Root)就是把一个数的各位数字相加,再将所得数的各位数字相加,直到所得数为一位数字为止.而这 ...
- linux下的代码比较工具
在linux下有很多不错的代码比较工具:meld.DiffMerge.xxdiff.diffuse.Kompare等... diff : 文件比较工具用于比较计算机上的文件的内容,找到他们之间相同与不 ...
- 转-decorators.xml的用法-http://blog.csdn.net/gavinloo/article/details/7458062
今天改前人做的项目,用struts2,spring,hibernate框架做的,对了,还有jQuery.我用jquery做异步请求到后台,生成json数据返回前台生成下拉输入框,请求到后台以后,成功生 ...
- rabiitmq集群完整安装
通过 Erlang 的分布式特性(通过 magic cookie 认证节点)进行 RabbitMQ 集群,各 RabbitMQ 服务为对等节点,即每个节点都提供服务给客户端连接,进行消息发送与接收. ...
- 记录思想分享知识编辑器 Markdown 编辑阅读器
web使用:实现网页客户端实时自动解析Markdown为HTML内容小小的展示:Cmd Markdown 编辑阅读器使用必要性:怎样引导新手使用 Markdown? - 写作
- oracle 科普
Schem定义 A schema is a collection of database objects (used by a user.). Schema objects are the logic ...