⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 duplication-of-side-effects.html

📁 gcc手册
💻 HTML
字号:
<html lang="en">

<head>

<title>The C Preprocessor</title>

<meta http-equiv="Content-Type" content="text/html">

<meta name="description" content="The C Preprocessor">

<meta name="generator" content="makeinfo 4.3">

<link href="http://www.gnu.org/software/texinfo/" rel="generator-home">

<!--

Copyright &copy; 1987, 1989, 1991, 1992, 1993, 1994, 1995, 1996,

1997, 1998, 1999, 2000, 2001, 2002, 2003

Free Software Foundation, Inc.



   <p>Permission is granted to copy, distribute and/or modify this document

under the terms of the GNU Free Documentation License, Version 1.1 or

any later version published by the Free Software Foundation.  A copy of

the license is included in the

section entitled "GNU Free Documentation License".



   <p>This manual contains no Invariant Sections.  The Front-Cover Texts are

(a) (see below), and the Back-Cover Texts are (b) (see below).



   <p>(a) The FSF's Front-Cover Text is:



   <p>A GNU Manual



   <p>(b) The FSF's Back-Cover Text is:



   <p>You have freedom to copy and modify this GNU Manual, like GNU

     software.  Copies published by the Free Software Foundation raise

     funds for GNU development. 

-->

</head>

<body>

<div class="node">

<p>

Node:<a name="Duplication%20of%20Side%20Effects">Duplication of Side Effects</a>,

Next:<a rel="next" accesskey="n" href="Self-Referential-Macros.html#Self-Referential%20Macros">Self-Referential Macros</a>,

Previous:<a rel="previous" accesskey="p" href="Swallowing-the-Semicolon.html#Swallowing%20the%20Semicolon">Swallowing the Semicolon</a>,

Up:<a rel="up" accesskey="u" href="Macro-Pitfalls.html#Macro%20Pitfalls">Macro Pitfalls</a>

<hr><br>

</div>



<h4 class="subsection">Duplication of Side Effects</h4>



   <p>Many C programs define a macro <code>min</code>, for "minimum", like this:



<pre class="example">     #define min(X, Y)  ((X) &lt; (Y) ? (X) : (Y))

     </pre>



   <p>When you use this macro with an argument containing a side effect,

as shown here,



<pre class="example">     next = min (x + y, foo (z));

     </pre>



<p>it expands as follows:



<pre class="example">     next = ((x + y) &lt; (foo (z)) ? (x + y) : (foo (z)));

     </pre>



<p>where <code>x + y</code> has been substituted for <code>X</code> and <code>foo (z)</code>

for <code>Y</code>.



   <p>The function <code>foo</code> is used only once in the statement as it appears

in the program, but the expression <code>foo (z)</code> has been substituted

twice into the macro expansion.  As a result, <code>foo</code> might be called

two times when the statement is executed.  If it has side effects or if

it takes a long time to compute, the results might not be what you

intended.  We say that <code>min</code> is an <dfn>unsafe</dfn> macro.



   <p>The best solution to this problem is to define <code>min</code> in a way that

computes the value of <code>foo (z)</code> only once.  The C language offers

no standard way to do this, but it can be done with GNU extensions as

follows:



<pre class="example">     #define min(X, Y)                \

     ({ typeof (X) x_ = (X);          \

        typeof (Y) y_ = (Y);          \

        (x_ &lt; y_) ? x_ : y_; })

     </pre>



   <p>The <code>({ ... })</code> notation produces a compound statement that

acts as an expression.  Its value is the value of its last statement. 

This permits us to define local variables and assign each argument to

one.  The local variables have underscores after their names to reduce

the risk of conflict with an identifier of wider scope (it is impossible

to avoid this entirely).  Now each argument is evaluated exactly once.



   <p>If you do not wish to use GNU C extensions, the only solution is to be

careful when <em>using</em> the macro <code>min</code>.  For example, you can

calculate the value of <code>foo (z)</code>, save it in a variable, and use

that variable in <code>min</code>:



<pre class="example">     #define min(X, Y)  ((X) &lt; (Y) ? (X) : (Y))

     ...

     {

       int tem = foo (z);

       next = min (x + y, tem);

     }

     </pre>



<p>(where we assume that <code>foo</code> returns type <code>int</code>).



   </body></html>



⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -