CodeForces - 7E Defining Macros【模拟】

【题目描述】
Most C/C++ programmers know about excellent opportunities that preprocessor #define directives give; but many know as well about the problems that can arise because of their careless use.

In this problem we consider the following model of #define constructions (also called macros). Each macro has its name and value. The generic syntax for declaring a macro is the following:

#define macro_name macro_value

After the macro has been declared, “macro_name” is replaced with “macro_value” each time it is met in the program (only the whole tokens can be replaced; i.e. “macro_name” is replaced only when it is surrounded by spaces or other non-alphabetic symbol). A “macro_value” within our model can only be an arithmetic expression consisting of variables, four arithmetic operations, brackets, and also the names of previously declared macros (in this case replacement is performed sequentially). The process of replacing macros with their values is called substitution.

One of the main problems arising while using macros — the situation when as a result of substitution we get an arithmetic expression with the changed order of calculation because of different priorities of the operations.

Let’s consider the following example. Say, we declared such a #define construction:

#define sum x + y

and further in the program the expression “2 * sum” is calculated. After macro substitution is performed we get “2 * x + y”, instead of intuitively expected “2 * (x + y)”.

Let’s call the situation “suspicious”, if after the macro substitution the order of calculation changes, falling outside the bounds of some macro. Thus, your task is to find out by the given set of #define definitions and the given expression if this expression is suspicious or not.

Let’s speak more formally. We should perform an ordinary macros substitution in the given expression. Moreover, we should perform a “safe” macros substitution in the expression, putting in brackets each macro value; after this, guided by arithmetic rules of brackets expansion, we can omit some of the brackets. If there exist a way to get an expression, absolutely coinciding with the expression that is the result of an ordinary substitution (character-by-character, but ignoring spaces), then this expression and the macros system are called correct, otherwise — suspicious.

Note that we consider the “/” operation as the usual mathematical division, not the integer division like in C/C++. That’s why, for example, in the expression “a*(b/c)” we can omit brackets to get the expression “a*b/c”.

【输入】
The first line contains the only number n (0 ≤ n ≤ 100) — the amount of #define constructions in the given program.

Then there follow n lines, each of them contains just one #define construction. Each construction has the following syntax:

#define name expression

where

name — the macro name,
expression — the expression with which the given macro will be replaced. An expression is a non-empty string, containing digits,names of variables, names of previously declared macros, round brackets and operational signs ±*/. It is guaranteed that the expression (before and after macros substitution) is a correct arithmetic expression, having no unary operations. The expression contains only non-negative integers, not exceeding 109.
All the names (#define constructions’ names and names of their arguments) are strings of case-sensitive Latin characters. It is guaranteed that the name of any variable is different from any #define construction.

Then, the last line contains an expression that you are to check. This expression is non-empty and satisfies the same limitations as the expressions in #define constructions.

The input lines may contain any number of spaces anywhere, providing these spaces do not break the word “define” or the names of constructions and variables. In particular, there can be any number of spaces before and after the “#” symbol.

The length of any line from the input file does not exceed 100 characters.

【输出】
Output “OK”, if the expression is correct according to the above given criterion, otherwise output “Suspicious”.

【样例输入】
1
#define sum x + y
1 * sum

【样例输出】
Suspicious

【样例输入】
1
#define sum (x + y)
sum - sum

【样例输出】
OK

【样例输入】
4
#define sum x + y
#define mul a * b
#define div a / b
#define expr sum + mul * div * mul
expr

【样例输出】
OK

【样例输入】
3
#define SumSafe (a+b)
#define DivUnsafe a/b
#define DenominatorUnsafe a*b
((SumSafe) + DivUnsafe/DivUnsafe + x/DenominatorUnsafe)

【样例输出】
Suspicious

本来以为是一道要写一天的大模拟
看了codeforces上大佬们写的代码差点哭出来,orz
改了改别人的代码发现除了输入输出别的地方一改就wa,orz

题目链接:https://codeforces.com/contest/7/problem/E

代码如下:

#include <bits/stdc++.h>
using namespace std;
static const int MAXN=100;
string tmp,name;
char s[MAXN+10];
int n;
map<string,int> mp;
int check(int l,int r)
{
    int t=0;
    for(int i=r-1;i>=l;i--)
    {
        t+=(s[i]=='(')-(s[i]==')');
        if(!t && (s[i]=='+' || s[i]=='-'))
        {
            int L=check(l,i),R=check(i+1,r);
            return L&&R&&(s[i]=='+' || R>1);
        }
    }
    t=0;
    for(int i=r-1;i>=l;i--)
    {
        t+=(s[i]=='(')-(s[i]==')');
        if(!t && (s[i]=='*' || s[i]=='/'))
        {
            int L=check(l,i),R=check(i+1,r);
            return (L>1&&R>1&&(s[i]=='*' || R>2))?2:0;
        }
    }
    if(s[l]=='(') return check(l+1,r-1)?3:0;
    string tmp(s+l,s+r);
    return mp.count(tmp)?mp[tmp]:3;
}
void change()
{
    getline(cin,tmp);
    int cnt=0;
    for(int i=0;i<tmp.size();i++)
        if(tmp[i]!=' ') s[cnt++]=tmp[i];
    s[cnt]='\0';
}
int main()
{
    cin>>n;
    for(int i=1;i<=n;i++)
    {
        cin>>tmp;
        if(tmp=="#") cin>>tmp;
        cin>>name;
        change();
        mp[name]=check(0,strlen(s));
    }
    change();
    cout<<(check(0,strlen(s))?"OK":"Suspicious")<<endl;
    return 0;
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值