高肆玖捌伍 发表于 2023-1-15 14:04:43

c语言问题求大佬解惑

#include<stdio.h>
#include<string.h>

#define TOUPPER(c) (((c)>='a'&&(c)<='z')?((c)-'a'+'A'):(c))

int main()
{
        char s;
        strcpy(s,"abcd");
        int i = 0;
        putchar(TOUPPER(s[++i]));
        printf("\n%d", i);
       
        return 0;
}
运行结果是:
D
3
为什么不是B和1呢{:5_99:}
拜托大家了

dolly_yos2 发表于 2023-1-15 14:18:44

因为宏是字符串替换,宏里的参数用了几次它的副作用就生效几次,所以这里 ++i 实际上执行了多次
原则是只要能不用宏就不用, inline 函数、 const 常量或者 enum hack 可以替代很多宏的功能并避免潜在的错误;宏函数的参数不要有副作用,或者保证幂等

高肆玖捌伍 发表于 2023-1-15 14:22:37

dolly_yos2 发表于 2023-1-15 14:18
因为宏是字符串替换,宏里的参数用了几次它的副作用就生效几次,所以这里 ++i 实际上执行了多次
原则是只 ...

感谢

人造人 发表于 2023-1-15 14:23:38

sh-5.1$ cat main.c
//#include<stdio.h>
//#include<string.h>

#define TOUPPER(c) (((c)>='a'&&(c)<='z')?((c)-'a'+'A'):(c))

int main()
{
      char s;
      strcpy(s,"abcd");
      int i = 0;
      putchar(TOUPPER(s[++i]));
      printf("\n%d", i);

      return 0;
}
sh-5.1$ gcc -E main.c
# 0 "main.c"
# 0 "<built-in>"
# 0 "<command-line>"
# 1 "/usr/include/stdc-predef.h" 1 3 4
# 0 "<command-line>" 2
# 1 "main.c"





int main()
{
      char s;
      strcpy(s,"abcd");
      int i = 0;
      putchar((((s[++i])>='a'&&(s[++i])<='z')?((s[++i])-'a'+'A'):(s[++i])));
      printf("\n%d", i);

      return 0;
}
sh-5.1$


看到了吗,看那个putchar
putchar(TOUPPER(s[++i]));
这个经过预处理后就变成下面这样了
putchar((((s[++i])>='a'&&(s[++i])<='z')?((s[++i])-'a'+'A'):(s[++i])));

宏替换就是单纯的替换

高肆玖捌伍 发表于 2023-1-15 14:32:18

人造人 发表于 2023-1-15 14:23
看到了吗,看那个putchar
putchar(TOUPPER(s[++i]));
这个经过预处理后就变成下面这样了


谢谢大佬,我明白了
页: [1]
查看完整版本: c语言问题求大佬解惑