git » homepage.git » master » tree

[master] / OUT / Lisp / CommonLispIteration.html

<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
<html>
<head>
	<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
	<title>Common Lisp iteration</title>
	<meta name='Generator' content='Zim 0.75.2'>
  <script async src="https://www.googletagmanager.com/gtag/js?id=G-XCMVL5K44X"></script>
  <script>
    window.dataLayer = window.dataLayer || [];
    function gtag(){dataLayer.push(arguments);}
    gtag('js', new Date());
    gtag('config', 'G-XCMVL5K44X');
  </script>
	<style type='text/css'>
    body {
        background-image: url();
    }
    div#main {
        margin: 40px auto;
        max-width: 800px;
        line-height: 1.4;
        font-size: 1.1em;
        padding: 1em;
        box-shadow: 0 4px 8px 0 rgba(0, 0, 0, 0.2), 0 6px 20px 0 rgba(0, 0, 0, 0.19);
        border-radius: 0.75em;
        background-color: white;
    }
    img[src*="float_right"] {
      float: right;
      border-radius: 0.75em;
    }
    img[src*="200px"] {
      width: 200px;
    }
		strike     { color: grey                }
		u          { text-decoration: none;
					 background-color: yellow   }
		tt         { color: #2e3436;            }
		pre        { color: #2e3436;
					 margin-left: 20px          }
		h1,h2,h3,h4,h5 {
				color: #cc3b12;
        margin: 0 auto;
    }
		p          { margin-top: 0              }
		span.zim-tag {
			color: #ce5c00;
		}
		div.zim-object {
			border-style:solid;
			border-width:1px;
		}
		.checked-box {list-style-image: url()}
		.xchecked-box {list-style-image: url()}
		.unchecked-box {list-style-image: url()}
		.migrated-box {list-style-image: url()}
		ul {list-style-image: none}
		/* ul rule needed to reset style for sub-bullets */
	</style>
</head>
<body>
  <div id="main">
<!-- Header -->
<div>
[ <a href='../Home.html'>Home</a> ]
[ <a href='../Index.html'>Index</a> ]
</div>

<hr />

<!-- Wiki content -->

<div class='pages'>
	<div class='heading'>
	<h1>Common Lisp iteration <a name='Lisp:CommonLispIteration'></a></h1>
	</div>

	<div class='content Lisp:CommonLispIteration'>
	<p>
Created Monday 23 November 2020
</p>
<br>
<p>
Each of the following definitions of a <a href="https://en.wikipedia.org/wiki/Factorial" title="factorial" class="https">factorial</a> function demonstrate a way to <a href="https://en.wikipedia.org/wiki/Iteration#Computing" title="iterate" class="https">iterate</a> in <a href="https://en.wikipedia.org/wiki/Common_Lisp" title="Common Lisp" class="https">Common Lisp</a>, with brief notes. I hope that by demonstrating many different ways that the same thing can be written, you can develop a sense for the character of the constructs afforded by the language, and of the variety of possible styles. Common Lisp is famously syntactically extensible via <a href="https://en.wikipedia.org/wiki/Common_Lisp#Macros" title="macros" class="https">macros</a>, so keep in mind that my examples are by no means the <i>only</i> ways to iterate.
</p>
<br>
<p>
For further reading on the iteration and control structures of Common Lisp, I heartily recommend:
</p>
<br>
<ul>
<li><a href="http://www.gigamonkeys.com/book/macros-standard-control-constructs.html" title="Chapter 7" class="http">Chapter 7</a> and <a href="http://www.gigamonkeys.com/book/loop-for-black-belts.html" title="Chapter 22" class="http">Chapter 22</a> of <a href="https://amzn.to/3nOWKa2" title="Practical Common Lisp" class="https">Practical Common Lisp</a> by Peter Siebel.</li>
<li>A reasonably-priced used copy of <a href="https://amzn.to/2UUTfm3" title="ANSI Common Lisp" class="https">ANSI Common Lisp</a> by Paul Graham.</li>
</ul>
<br>
<p>
<i>Note: several of the examples return nonsensical results for negative inputs. The addition of <tt>(assert (not (minusp n)) </tt>or similar is a good idea, but I have omitted it here for clarity.</i>
</p>
<br>
<h2>DOTIMES<a id="dotimes" class="h_anchor"></a></h2>
<pre>
(defun factorial-dotimes (n &amp;aux (prod 1))
  (dotimes (i n prod)
    (setq prod (* prod (1+ i)))))
</pre>
<br>
<ul>
<li><a href="http://www.lispworks.com/documentation/HyperSpec/Body/03_dae.htm" title="&amp;aux lambda list keyword" class="http"><tt>&amp;aux</tt> lambda list keyword</a> names a local variable <tt>prod</tt>. <a href="http://www.lispworks.com/documentation/HyperSpec/Body/s_let_l.htm" title="LET" class="http"><tt>LET</tt></a> could also be used for this purpose, but at the cost of more indentation.</li>
<li><a href="http://www.lispworks.com/documentation/lw50/CLHS/Body/m_dotime.htm" title="DOTIMES" class="http"><tt>DOTIMES</tt></a> binds <tt>i</tt> successively from 0 to 1-n and finally evaluates to <tt>prod</tt>.</li>
</ul>
<br>
<h2>DO<a id="do" class="h_anchor"></a></h2>
<pre>
(defun factorial-do (n)
  (do ((i 1 (1+ i))
       (prod 1 (* prod i)))
      ((&gt; i n) prod)))
</pre>
<br>
<ul>
<li><a href="http://www.lispworks.com/documentation/lw50/CLHS/Body/m_do_do.htm" title="DO" class="http"><tt>DO</tt></a> binds <tt>i</tt> to 1 and then to (1+ i) in subsequent iterations. <tt>prod</tt> is bound first to 1 and then to <tt>(* prod i)</tt> in subsequent iterations.</li>
<li>When the test clause <tt>(&gt; i n)</tt> becomes true, <tt>prod</tt> is returned. Contrast with the test clause of <tt>for</tt> loops in other languages, which terminate the loop when they become <i>false</i>.</li>
<li>I like the way Paul Graham explains <tt>DO </tt>and<tt> DO*</tt> in <a href="https://amzn.to/2UUTfm3" title="ANSI Common Lisp" class="https">ANSI Common Lisp</a>.</li>
</ul>
<br>
<h2>LOOP<a id="loop" class="h_anchor"></a></h2>
<pre>
(defun factorial-loop (n)
  (loop
     for i from 1 to n
     for prod = 1 then (* prod i)
     finally (return prod)))
</pre>
<br>
<ul>
<li><tt>i</tt> is bound from 1 to <tt>n</tt> inclusive.</li>
<li><tt>prod</tt> is bound to 1 and then <tt>(* prod i)</tt> in subsequent iterations in a manner similar to <tt>DO</tt>.</li>
<li>In the <tt>finally</tt> clause, <tt>prod</tt> is returned by <a href="http://www.lispworks.com/documentation/lw60/CLHS/Body/m_return.htm#return" title="RETURN" class="http"><tt>RETURN</tt></a> once iteration is complete. The <a href="http://www.lispworks.com/documentation/lw60/CLHS/Body/s_block.htm#block" title="BLOCK" class="http"><tt>BLOCK</tt></a> named NIL established by <tt>LOOP</tt> is the point of return.</li>
<li><a href="http://www.lispworks.com/documentation/lw50/CLHS/Body/m_loop.htm" title="LOOP" class="http"><tt>LOOP</tt></a> supports a comprehensive iteration and accumulation <a href="https://en.wikipedia.org/wiki/Domain-specific_language" title="DSL" class="https">DSL</a>. <a href="http://www.gigamonkeys.com/book/loop-for-black-belts.html" title="Chapter 22" class="http">Chapter 22</a> of <a href="https://amzn.to/3nOWKa2" title="Practical Common Lisp" class="https">Practical Common Lisp</a> offers a great introduction.</li>
</ul>
<br>
<p>
The preceding example demonstrates the "extended" form of <tt>LOOP</tt>. There's also "simple" form:
</p>
<br>
<pre>
(defun factorial-simple-loop (n &amp;aux (i 0) (prod 1))
  (loop
    (when (eql i n)
      (return prod))
    (setq prod (* prod (incf i)))))
</pre>
<br>
<h2>Recursion<a id="recursion" class="h_anchor"></a></h2>
<pre>
(defun factorial-recursive (n)
  (if (zerop n)
      1
      (* n (factorial-recursive (1- n)))))
</pre>
<br>
<ul>
<li><tt>FACTORIAL-RECURSIVE</tt> calls itself, but when <tt>n</tt> exceeds the maximum stack size supported by the implementation, an error is signaled.</li>
</ul>
<br>
<pre>
(defun factorial-tail-recursive (n)
  (labels ((recur (n prod)
             (if (zerop n)
                 prod
                 (recur (1- n) (* n prod)))))
    (recur n 1)))
</pre>
<br>
<ul>
<li><tt>FACTORIAL-TAIL-RECURSIVE </tt>does not call itself directly.</li>
<li>Instead, it defines with <a href="http://www.lispworks.com/documentation/HyperSpec/Body/s_flet_.htm" title="LABELS" class="http"><tt>LABELS</tt></a> an internal and recursive helper function, <tt>recur</tt>.</li>
<li>recur <a href="https://en.wikipedia.org/wiki/Tail_call" title="calls itself in tail position" class="https">calls itself in tail position</a> and the stack never overflows in implementations that implement tail-call elimination.</li>
</ul>
<br>
<pre>
(defun factorial-tail-recursive-opt (n &amp;optional (prod 1))
  (if (zerop n)
      prod
    (factorial-tail-recursive-opt (1- n) (* n prod))))
</pre>
<br>
<br>
<ul>
<li><tt>FACTORIAL-TAIL-RECURSIVE-OPT</tt> is also tail recursive, but uses the <tt>&amp;OPTIONAL</tt> lambda list keyword to maintain <tt>prod</tt> across iterations. This approach has the downside of exposing <tt>prod</tt> as part of the public interface of the function. Arguably, <tt>prod</tt> is an implementation detail, best kept internal.</li>
</ul>
<br>
<h2>PROG<a id="prog" class="h_anchor"></a></h2>
<pre>
(defun factorial-prog (n)
  (prog ((i 0) (prod 1))
   begin
   (when (eql i n)
     (return prod))
   (setq prod (* prod (incf i)))
   (go begin)))
</pre>
<br>
<ul>
<li>PROG supports both declaring local lexical variables (<tt>i</tt> and <tt>prod</tt>) and naming GO tags (<tt>begin</tt>).</li>
<li><tt>begin</tt> names a label within the <i>implicit <tt>TAGBODY</tt></i> enclosed by <tt>PROG</tt> that may be jumped to.</li>
<li><a href="http://www.lispworks.com/documentation/HyperSpec/Body/m_when_.htm" title="WHEN" class="http"><tt>WHEN</tt></a> <tt>i</tt> is <a href="http://www.lispworks.com/documentation/HyperSpec/Body/f_eql.htm" title="EQL" class="http"><tt>EQL</tt></a> to <tt>n</tt>, <a href="http://www.lispworks.com/documentation/lw60/CLHS/Body/s_ret_fr.htm" title="RETURN" class="http"><tt>RETURN</tt></a> returns <tt>prod</tt>.</li>
<li><a href="http://www.lispworks.com/documentation/HyperSpec/Body/s_go.htm" title="GO" class="http"><tt>GO</tt></a> jumps to <tt>begin</tt>.</li>
</ul>
<br>
<h2>TAGBODY<a id="tagbody" class="h_anchor"></a></h2>
<pre>
(defun factorial-tagbody (n &amp;aux (i 0) (prod 1))
  (tagbody
     begin
     (when (eql i n)
       (return-from factorial-tagbody prod))
     (setq prod (* prod (incf i)))
     (go begin)))
</pre>
<br>
<ul>
<li><a href="http://www.lispworks.com/documentation/HyperSpec/Body/s_tagbod.htm" title="TAGBODY" class="http"><tt>TAGBODY</tt></a> is the most general but also the lowest-level and most verbose iteration construct.</li>
<li><a href="http://www.lispworks.com/documentation/HyperSpec/Body/03_dae.htm" title="&amp;aux lambda list keyword" class="http"><tt>&amp;aux</tt> lambda list keyword</a> names local variables <tt>i</tt> and <tt>prod</tt>, initializing them to 0 and 1, respectively.</li>
<li><a href="http://www.lispworks.com/documentation/HyperSpec/Body/m_when_.htm" title="WHEN" class="http"><tt>WHEN</tt></a> <tt>i</tt> is <a href="http://www.lispworks.com/documentation/HyperSpec/Body/f_eql.htm" title="EQL" class="http"><tt>EQL</tt></a> to <tt>n</tt>, <a href="http://www.lispworks.com/documentation/lw60/CLHS/Body/s_ret_fr.htm" title="RETURN-FROM" class="http"><tt>RETURN-FROM</tt></a> returns <tt>prod</tt> from the <a href="http://www.lispworks.com/documentation/lw60/CLHS/Body/s_block.htm#block" title="BLOCK" class="http"><tt>BLOCK</tt></a> named after the function by <a href="http://www.lispworks.com/documentation/lw60/CLHS/Body/m_defun.htm" title="DEFUN" class="http"><tt>DEFUN</tt></a>.</li>
<li><a href="http://www.lispworks.com/documentation/HyperSpec/Body/s_go.htm" title="GO" class="http"><tt>GO</tt></a> jumps to <tt>begin</tt>.</li>
</ul>

	</div>

	<br />

	<div class='page-footer'>
		<b>Backlinks:</b>

		<a href='../Home.html'>Home</a>

		
		

		<a href='./CommonLisp.html'>Lisp:CommonLisp</a>

		<br /><br />

	</div>

	

</div>

</div id="main">
</body>
</html>