Вы не можете выбрать более 25 тем Темы должны начинаться с буквы или цифры, могут содержать дефисы(-) и должны содержать не более 35 символов.

156 lines
6.8KB

  1. <!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
  2. <html>
  3. <!-- Copyright (C) 1987-2020 Free Software Foundation, Inc.
  4. Permission is granted to copy, distribute and/or modify this document
  5. under the terms of the GNU Free Documentation License, Version 1.3 or
  6. any later version published by the Free Software Foundation. A copy of
  7. the license is included in the
  8. section entitled "GNU Free Documentation License".
  9. This manual contains no Invariant Sections. The Front-Cover Texts are
  10. (a) (see below), and the Back-Cover Texts are (b) (see below).
  11. (a) The FSF's Front-Cover Text is:
  12. A GNU Manual
  13. (b) The FSF's Back-Cover Text is:
  14. You have freedom to copy and modify this GNU Manual, like GNU
  15. software. Copies published by the Free Software Foundation raise
  16. funds for GNU development. -->
  17. <!-- Created by GNU Texinfo 6.5, http://www.gnu.org/software/texinfo/ -->
  18. <head>
  19. <meta http-equiv="Content-Type" content="text/html; charset=utf-8">
  20. <title>Duplication of Side Effects (The C Preprocessor)</title>
  21. <meta name="description" content="Duplication of Side Effects (The C Preprocessor)">
  22. <meta name="keywords" content="Duplication of Side Effects (The C Preprocessor)">
  23. <meta name="resource-type" content="document">
  24. <meta name="distribution" content="global">
  25. <meta name="Generator" content="makeinfo">
  26. <link href="index.html#Top" rel="start" title="Top">
  27. <link href="Index-of-Directives.html#Index-of-Directives" rel="index" title="Index of Directives">
  28. <link href="index.html#SEC_Contents" rel="contents" title="Table of Contents">
  29. <link href="Macro-Pitfalls.html#Macro-Pitfalls" rel="up" title="Macro Pitfalls">
  30. <link href="Self_002dReferential-Macros.html#Self_002dReferential-Macros" rel="next" title="Self-Referential Macros">
  31. <link href="Swallowing-the-Semicolon.html#Swallowing-the-Semicolon" rel="prev" title="Swallowing the Semicolon">
  32. <style type="text/css">
  33. <!--
  34. a.summary-letter {text-decoration: none}
  35. blockquote.indentedblock {margin-right: 0em}
  36. blockquote.smallindentedblock {margin-right: 0em; font-size: smaller}
  37. blockquote.smallquotation {font-size: smaller}
  38. div.display {margin-left: 3.2em}
  39. div.example {margin-left: 3.2em}
  40. div.lisp {margin-left: 3.2em}
  41. div.smalldisplay {margin-left: 3.2em}
  42. div.smallexample {margin-left: 3.2em}
  43. div.smalllisp {margin-left: 3.2em}
  44. kbd {font-style: oblique}
  45. pre.display {font-family: inherit}
  46. pre.format {font-family: inherit}
  47. pre.menu-comment {font-family: serif}
  48. pre.menu-preformatted {font-family: serif}
  49. pre.smalldisplay {font-family: inherit; font-size: smaller}
  50. pre.smallexample {font-size: smaller}
  51. pre.smallformat {font-family: inherit; font-size: smaller}
  52. pre.smalllisp {font-size: smaller}
  53. span.nolinebreak {white-space: nowrap}
  54. span.roman {font-family: initial; font-weight: normal}
  55. span.sansserif {font-family: sans-serif; font-weight: normal}
  56. ul.no-bullet {list-style: none}
  57. -->
  58. </style>
  59. </head>
  60. <body lang="en">
  61. <a name="Duplication-of-Side-Effects"></a>
  62. <div class="header">
  63. <p>
  64. Next: <a href="Self_002dReferential-Macros.html#Self_002dReferential-Macros" accesskey="n" rel="next">Self-Referential Macros</a>, Previous: <a href="Swallowing-the-Semicolon.html#Swallowing-the-Semicolon" accesskey="p" rel="prev">Swallowing the Semicolon</a>, Up: <a href="Macro-Pitfalls.html#Macro-Pitfalls" accesskey="u" rel="up">Macro Pitfalls</a> &nbsp; [<a href="index.html#SEC_Contents" title="Table of contents" rel="contents">Contents</a>][<a href="Index-of-Directives.html#Index-of-Directives" title="Index" rel="index">Index</a>]</p>
  65. </div>
  66. <hr>
  67. <a name="Duplication-of-Side-Effects-1"></a>
  68. <h4 class="subsection">3.10.4 Duplication of Side Effects</h4>
  69. <a name="index-side-effects-_0028in-macro-arguments_0029"></a>
  70. <a name="index-unsafe-macros"></a>
  71. <p>Many C programs define a macro <code>min</code>, for &ldquo;minimum&rdquo;, like this:
  72. </p>
  73. <div class="smallexample">
  74. <pre class="smallexample">#define min(X, Y) ((X) &lt; (Y) ? (X) : (Y))
  75. </pre></div>
  76. <p>When you use this macro with an argument containing a side effect,
  77. as shown here,
  78. </p>
  79. <div class="smallexample">
  80. <pre class="smallexample">next = min (x + y, foo (z));
  81. </pre></div>
  82. <p>it expands as follows:
  83. </p>
  84. <div class="smallexample">
  85. <pre class="smallexample">next = ((x + y) &lt; (foo (z)) ? (x + y) : (foo (z)));
  86. </pre></div>
  87. <p>where <code>x + y</code> has been substituted for <code>X</code> and <code>foo (z)</code>
  88. for <code>Y</code>.
  89. </p>
  90. <p>The function <code>foo</code> is used only once in the statement as it appears
  91. in the program, but the expression <code>foo (z)</code> has been substituted
  92. twice into the macro expansion. As a result, <code>foo</code> might be called
  93. two times when the statement is executed. If it has side effects or if
  94. it takes a long time to compute, the results might not be what you
  95. intended. We say that <code>min</code> is an <em>unsafe</em> macro.
  96. </p>
  97. <p>The best solution to this problem is to define <code>min</code> in a way that
  98. computes the value of <code>foo (z)</code> only once. The C language offers
  99. no standard way to do this, but it can be done with GNU extensions as
  100. follows:
  101. </p>
  102. <div class="smallexample">
  103. <pre class="smallexample">#define min(X, Y) \
  104. ({ typeof (X) x_ = (X); \
  105. typeof (Y) y_ = (Y); \
  106. (x_ &lt; y_) ? x_ : y_; })
  107. </pre></div>
  108. <p>The &lsquo;<samp>({ &hellip; })</samp>&rsquo; notation produces a compound statement that
  109. acts as an expression. Its value is the value of its last statement.
  110. This permits us to define local variables and assign each argument to
  111. one. The local variables have underscores after their names to reduce
  112. the risk of conflict with an identifier of wider scope (it is impossible
  113. to avoid this entirely). Now each argument is evaluated exactly once.
  114. </p>
  115. <p>If you do not wish to use GNU C extensions, the only solution is to be
  116. careful when <em>using</em> the macro <code>min</code>. For example, you can
  117. calculate the value of <code>foo (z)</code>, save it in a variable, and use
  118. that variable in <code>min</code>:
  119. </p>
  120. <div class="smallexample">
  121. <pre class="smallexample">#define min(X, Y) ((X) &lt; (Y) ? (X) : (Y))
  122. &hellip;
  123. {
  124. int tem = foo (z);
  125. next = min (x + y, tem);
  126. }
  127. </pre></div>
  128. <p>(where we assume that <code>foo</code> returns type <code>int</code>).
  129. </p>
  130. <hr>
  131. <div class="header">
  132. <p>
  133. Next: <a href="Self_002dReferential-Macros.html#Self_002dReferential-Macros" accesskey="n" rel="next">Self-Referential Macros</a>, Previous: <a href="Swallowing-the-Semicolon.html#Swallowing-the-Semicolon" accesskey="p" rel="prev">Swallowing the Semicolon</a>, Up: <a href="Macro-Pitfalls.html#Macro-Pitfalls" accesskey="u" rel="up">Macro Pitfalls</a> &nbsp; [<a href="index.html#SEC_Contents" title="Table of contents" rel="contents">Contents</a>][<a href="Index-of-Directives.html#Index-of-Directives" title="Index" rel="index">Index</a>]</p>
  134. </div>
  135. </body>
  136. </html>