zoukankan      html  css  js  c++  java
  • #define is unsafe

    #define is unsafe

    Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others)
    Total Submission(s): 66 Accepted Submission(s): 52
     
    Problem Description
    Have you used #define in C/C++ code like the code below?

    #include <stdio.h>
    #define MAX(a , b) ((a) > (b) ? (a) : (b))
    int main()
    {
      printf("%d " , MAX(2 + 3 , 4));
      return 0;
    }

    Run the code and get an output: 5, right?
    You may think it is equal to this code:

    #include <stdio.h>
    int max(a , b) {  return ((a) > (b) ? (a) : (b));  }
    int main()
    {
      printf("%d " , max(2 + 3 , 4));
      return 0;
    }

    But they aren't.Though they do produce the same anwser , they work in two different ways.
    The first code, just replace the MAX(2 + 3 , 4) with ((2 + 3) > (4) ? (2 + 3) : 4), which calculates (2 + 3) twice.
    While the second calculates (2 + 3) first, and send the value (5 , 4) to function max(a , b) , which calculates (2 + 3) only once.

    What about MAX( MAX(1+2,2) , 3 ) ?
    Remember "replace".
    First replace: MAX( (1 + 2) > 2 ? (1 + 2) : 2 , 3)
    Second replace: ( ( ( 1 + 2 ) > 2 ? ( 1 + 2 ) : 2 ) > 3 ? ( ( 1 + 2 ) > 2 ? ( 1 + 2 ) : 2 ) : 3).
    The code may calculate the same expression many times like ( 1 + 2 ) above.
    So #define isn't good.In this problem,I'll give you some strings, tell me the result and how many additions(加法) are computed.
     
    Input
    The first line is an integer T(T<=40) indicating case number.
    The next T lines each has a string(no longer than 1000), with MAX(a,b), digits, '+' only(Yes, there're no other characters).
    In MAX(a,b), a and b may be a string with MAX(c,d), digits, '+'.See the sample and things will be clearer.
     
    Output
    For each case, output two integers in a line separated by a single space.Integers in output won't exceed 1000000.
     
    Sample Input
    6
    MAX(1,0)
    1+MAX(1,0)
    MAX(2+1,3)
    MAX(4,2+2)
    MAX(1+1,2)+MAX(2,3)
    MAX(MAX(1+2,3),MAX(4+5+6,MAX(7+8,9)))+MAX(10,MAX(MAX(11,12),13))
     
    Sample Output
    1 0
    2 1
    3 1
    4 2
    5 2
    28 14
     
    Author
    madfrog
     
    Source
    HDU2010省赛集训队选拔赛(校内赛)
     
    Recommend
    lcy
     
    /*
    题意:模拟题,给你一行只有MAX函数,和加法运算的表达式,输出表达式的值,并且统计在本质MAX运算中+运算的次数
    
    初步思路:一行表达式中有用的有 (:表示一个运算开始了 , +:表示有一个加法运算, ):表示一个运算结束了
        
    */
    #include<bits/stdc++.h>
    using namespace std;
    struct node{
        int a;//用来存储数字
        int res;//用来存储加法运算的次数
        node(){};
        node(int b,int c){
            a=b;
            res=c;
        }
    };
    stack<node>Q;//用来存储数字
    stack<char>oper;//用来存储运算符
    node x1,x2;
    int n;
    char str[1010];
    int main(){
        // freopen("in.txt","r",stdin);
        scanf("%d",&n);
        getchar();
        while(n--){
            gets(str);
            //cout<<str<<endl;
            for(int i=0;i<strlen(str);i++){
                if(str[i]>='0'&&str[i]<='9'){//如果当前位置的是数字的话,直接将他压进栈中
                    int cur=str[i]-'0';
                    i++;
                    while(str[i]>='0'&&str[i]<='9'){
                        cur=cur*10+str[i]-'0';
                        i++;
                    }
                    i--;//这位不是数字了但是不能继续往前走,要让for循环来承担这个任务
                    Q.push(node(cur,0));//将这个数字装进栈中,并且初始化经过乘法运算的次数为零
                }else if(str[i]=='+'||str[i]=='('){
                    oper.push(str[i]);
                }else if(str[i]==','){//如果遇见','就要将','左边的加法运算进行计算之后重新压到栈中
                    while(!oper.empty()&&oper.top()=='+'){//提取到一个加号就要运算一次
                        x1=Q.top();
                        Q.pop();
                        x2=Q.top();
                        Q.pop();
                        x1.a+=x2.a;
                        x1.res+=(x2.res+1);//这地方将max函数展开看就知道为什么加法运算是累加了
                        Q.push(x1);
                        oper.pop();//将这个运算符出栈
                    }
                }else if(str[i]==')'){//遇到一个外括号肯定是一个max运算结束了
                    //将这之前的加法运算全部算完
                    while(!oper.empty()&&oper.top()=='+'){//提取到一个加号就要运算一次
                        x1=Q.top();
                        Q.pop();
                        x2=Q.top();
                        Q.pop();
                        x1.a+=x2.a;
                        x1.res+=(x2.res+1);//这地方将max函数展开看就知道为什么加法运算是累加了
                        Q.push(x1);
                        oper.pop();//将这个运算符出栈
                    }
                    oper.pop();
                    x2=Q.top();
                    Q.pop();
                    x1=Q.top();//后取的才是第一个
                    Q.pop();
                    if(x1.a>x2.a){//说明x1的部分要运算两次
                        x1.res=x1.res*2+x2.res;
                        Q.push(x1);
                    }else{//否则的话就是x2的部分要运算两次
                        x1.res=x1.res+x2.res*2;
                        x1.a=x2.a;
                        Q.push(x1);
                    }
                }
            }
            //将所有的操作进行之后,栈中如果还有运算,只能是MAX函数之间的加法运算
            while(!oper.empty()){
                x1=Q.top();
                Q.pop();
                x2=Q.top();
                Q.pop();
                x1.a+=x2.a;
                x1.res+=(x2.res+1);
                Q.push(x1);
                oper.pop();
            }
            printf("%d %d
    ",Q.top().a,Q.top().res);
            Q.pop();//将最后一个元素出栈
        }
        return 0;
    }
  • 相关阅读:
    XML 文档(1, 2)中有错误:不应有 <xml xmlns=''>
    工厂模式、控制反转及依赖注入
    [译]新的CCSDS图像压缩推荐标准
    关于C#的静态类和静态构造函数
    c++友元函数与友元类
    malloc/free 与 new/delete 比较
    C++ main函数命令行参数使用
    进程和线程的区别
    堆和栈的区别
    java与C++的区别
  • 原文地址:https://www.cnblogs.com/wuwangchuxin0924/p/6399849.html
Copyright © 2011-2022 走看看