免费注册 查看新帖 |

Chinaunix

  平台 论坛 博客 文库
最近访问板块 发新帖
查看: 940 | 回复: 0
打印 上一主题 下一主题

Why are there sometimes meaningless do/while and i [复制链接]

论坛徽章:
0
跳转到指定楼层
1 [收藏(0)] [报告]
发表于 2009-12-01 11:27 |只看该作者 |倒序浏览

http://stackoverflow.com/questions/154136/why-are-there-sometimes-meaningless-dowhile-and-ifelse-statements-in-cc-macros

Answer:

The do ... while and if ... else are
there to make it so that a semicolon after your macro always means the
same thing. Let's say you had something like your second macro.
#define BAR(X) f(x); g(x)
Now if you were to use BAR(X); in an if ... else statement, where the bodies of the if statement were not wrapped in curly brackets, you'd get a bad surprise.
if (corge)
  BAR(corge);
else
  gralt();
The above code would expand into
if (corge)
  f(corge); g(corge);
else
  gralt();
which
is syntactically incorrect, as the else is no longer associated with
the if. It doesn't help to wrap things in curly braces within the
macro, because the following is also syntactically incorrect.
if (corge)
  {f(corge); g(corge);};
else
  gralt();
There
are two ways of fixing the problem. The first is to use a comma to
sequence statements within the macro without robbing it of its ability
to act like an expression.
#define BAR(X) f(X), g(X)
The above version of bar BAR expands the above code into what follows, which is syntactically correct.
if (corge)
  f(corge), g(corge);
else
  gralt();
This doesn't work if instead of f(X) you
have a more complicated body of code that needs to go in its own block,
say for example to declare local variables. In the most general case
the solution is to use something like do ... while to cause the macro to be a single statement that takes a semicolon without confusion.
#define BAR(X) do { \
  int i = f(X); \
  if (i > 4) g(i); \
} while (0)
You don't have to use do ... while, you could cook up something with if ... else as well, although when if ... else expands inside of an if ... else it leads to a "dangling else", which could make an existing dangling else problem even harder to find, as in the following code.
if (corge)
  if (1) { f(corge); g(corge); } else;
else
  gralt();
The
point is to use up the semicolon in contexts where a dangling semicolon
is erroneous. Of course, it could (and probably should) be argued at
this point that it would be better to declare BAR as an actual function, not a macro.
In summary, the do ... while is
there to work around the shortcomings of the C preprocessor. When those
C style guides tell you to lay off the C preprocessor, this is the kind
of thing they're worried about.
               
               
               

本文来自ChinaUnix博客,如果查看原文请点:http://blog.chinaunix.net/u1/34411/showart_2108107.html
您需要登录后才可以回帖 登录 | 注册

本版积分规则 发表回复

  

北京盛拓优讯信息技术有限公司. 版权所有 京ICP备16024965号-6 北京市公安局海淀分局网监中心备案编号:11010802020122 niuxiaotong@pcpop.com 17352615567
未成年举报专区
中国互联网协会会员  联系我们:huangweiwei@itpub.net
感谢所有关心和支持过ChinaUnix的朋友们 转载本站内容请注明原作者名及出处

清除 Cookies - ChinaUnix - Archiver - WAP - TOP