<?php
include_once $_SERVER['DOCUMENT_ROOT'] . '/include/shared-manual.inc';
$TOC = array();
$TOC_DEPRECATED = array();
$PARENTS = array();
include_once dirname(__FILE__) ."/toc/language.generators.inc";
$setup = array (
  'home' => 
  array (
    0 => 'index.php',
    1 => 'PHP Manual',
  ),
  'head' => 
  array (
    0 => 'UTF-8',
    1 => 'en',
  ),
  'this' => 
  array (
    0 => 'language.generators.comparison.php',
    1 => 'Comparing generators with Iterator objects',
    2 => 'Comparing generators with Iterator objects',
  ),
  'up' => 
  array (
    0 => 'language.generators.php',
    1 => 'Generators',
  ),
  'prev' => 
  array (
    0 => 'language.generators.syntax.php',
    1 => 'Generator syntax',
  ),
  'next' => 
  array (
    0 => 'language.attributes.php',
    1 => 'Attributes',
  ),
  'alternatives' => 
  array (
  ),
  'source' => 
  array (
    'lang' => 'en',
    'path' => 'language/generators.xml',
  ),
  'history' => 
  array (
  ),
);
$setup["toc"] = $TOC;
$setup["toc_deprecated"] = $TOC_DEPRECATED;
$setup["parents"] = $PARENTS;
manual_setup($setup);

contributors($setup);

?>
<div id="language.generators.comparison" class="sect1">
  <h2 class="title">Comparing generators with <span class="classname"><a href="class.iterator.php" class="classname">Iterator</a></span> objects</h2>

  <p class="para">
   The primary advantage of generators is their simplicity. Much less
   boilerplate code has to be written compared to implementing an
   <span class="classname"><a href="class.iterator.php" class="classname">Iterator</a></span> class, and the code is generally much more
   readable. For example, the following function and class are equivalent:
  </p>

  <div class="informalexample">
   <div class="example-contents">
<div class="annotation-non-interactive phpcode"><pre><code style="color: #000000"><span style="color: #0000BB">&lt;?php
</span><span style="color: #007700">function </span><span style="color: #0000BB">getLinesFromFile</span><span style="color: #007700">(</span><span style="color: #0000BB">$fileName</span><span style="color: #007700">) {
    if (!</span><span style="color: #0000BB">$fileHandle </span><span style="color: #007700">= </span><span style="color: #0000BB">fopen</span><span style="color: #007700">(</span><span style="color: #0000BB">$fileName</span><span style="color: #007700">, </span><span style="color: #DD0000">'r'</span><span style="color: #007700">)) {
        return;
    }

    while (</span><span style="color: #0000BB">false </span><span style="color: #007700">!== </span><span style="color: #0000BB">$line </span><span style="color: #007700">= </span><span style="color: #0000BB">fgets</span><span style="color: #007700">(</span><span style="color: #0000BB">$fileHandle</span><span style="color: #007700">)) {
        yield </span><span style="color: #0000BB">$line</span><span style="color: #007700">;
    }

    </span><span style="color: #0000BB">fclose</span><span style="color: #007700">(</span><span style="color: #0000BB">$fileHandle</span><span style="color: #007700">);
}

</span><span style="color: #FF8000">// versus...

</span><span style="color: #007700">class </span><span style="color: #0000BB">LineIterator </span><span style="color: #007700">implements </span><span style="color: #0000BB">Iterator </span><span style="color: #007700">{
    protected </span><span style="color: #0000BB">$fileHandle</span><span style="color: #007700">;

    protected </span><span style="color: #0000BB">$line</span><span style="color: #007700">;
    protected </span><span style="color: #0000BB">$i</span><span style="color: #007700">;

    public function </span><span style="color: #0000BB">__construct</span><span style="color: #007700">(</span><span style="color: #0000BB">$fileName</span><span style="color: #007700">) {
        if (!</span><span style="color: #0000BB">$this</span><span style="color: #007700">-&gt;</span><span style="color: #0000BB">fileHandle </span><span style="color: #007700">= </span><span style="color: #0000BB">fopen</span><span style="color: #007700">(</span><span style="color: #0000BB">$fileName</span><span style="color: #007700">, </span><span style="color: #DD0000">'r'</span><span style="color: #007700">)) {
            throw new </span><span style="color: #0000BB">RuntimeException</span><span style="color: #007700">(</span><span style="color: #DD0000">'Couldn\'t open file "' </span><span style="color: #007700">. </span><span style="color: #0000BB">$fileName </span><span style="color: #007700">. </span><span style="color: #DD0000">'"'</span><span style="color: #007700">);
        }
    }

    public function </span><span style="color: #0000BB">rewind</span><span style="color: #007700">() {
        </span><span style="color: #0000BB">fseek</span><span style="color: #007700">(</span><span style="color: #0000BB">$this</span><span style="color: #007700">-&gt;</span><span style="color: #0000BB">fileHandle</span><span style="color: #007700">, </span><span style="color: #0000BB">0</span><span style="color: #007700">);
        </span><span style="color: #0000BB">$this</span><span style="color: #007700">-&gt;</span><span style="color: #0000BB">line </span><span style="color: #007700">= </span><span style="color: #0000BB">fgets</span><span style="color: #007700">(</span><span style="color: #0000BB">$this</span><span style="color: #007700">-&gt;</span><span style="color: #0000BB">fileHandle</span><span style="color: #007700">);
        </span><span style="color: #0000BB">$this</span><span style="color: #007700">-&gt;</span><span style="color: #0000BB">i </span><span style="color: #007700">= </span><span style="color: #0000BB">0</span><span style="color: #007700">;
    }

    public function </span><span style="color: #0000BB">valid</span><span style="color: #007700">() {
        return </span><span style="color: #0000BB">false </span><span style="color: #007700">!== </span><span style="color: #0000BB">$this</span><span style="color: #007700">-&gt;</span><span style="color: #0000BB">line</span><span style="color: #007700">;
    }

    public function </span><span style="color: #0000BB">current</span><span style="color: #007700">() {
        return </span><span style="color: #0000BB">$this</span><span style="color: #007700">-&gt;</span><span style="color: #0000BB">line</span><span style="color: #007700">;
    }

    public function </span><span style="color: #0000BB">key</span><span style="color: #007700">() {
        return </span><span style="color: #0000BB">$this</span><span style="color: #007700">-&gt;</span><span style="color: #0000BB">i</span><span style="color: #007700">;
    }

    public function </span><span style="color: #0000BB">next</span><span style="color: #007700">() {
        if (</span><span style="color: #0000BB">false </span><span style="color: #007700">!== </span><span style="color: #0000BB">$this</span><span style="color: #007700">-&gt;</span><span style="color: #0000BB">line</span><span style="color: #007700">) {
            </span><span style="color: #0000BB">$this</span><span style="color: #007700">-&gt;</span><span style="color: #0000BB">line </span><span style="color: #007700">= </span><span style="color: #0000BB">fgets</span><span style="color: #007700">(</span><span style="color: #0000BB">$this</span><span style="color: #007700">-&gt;</span><span style="color: #0000BB">fileHandle</span><span style="color: #007700">);
            </span><span style="color: #0000BB">$this</span><span style="color: #007700">-&gt;</span><span style="color: #0000BB">i</span><span style="color: #007700">++;
        }
    }

    public function </span><span style="color: #0000BB">__destruct</span><span style="color: #007700">() {
        </span><span style="color: #0000BB">fclose</span><span style="color: #007700">(</span><span style="color: #0000BB">$this</span><span style="color: #007700">-&gt;</span><span style="color: #0000BB">fileHandle</span><span style="color: #007700">);
    }
}</span></code></pre></div>
   </div>

  </div>

  <p class="para">
   This flexibility does come at a cost, however: generators are forward-only
   iterators, and cannot be rewound once iteration has started. This also
   means that the same generator can&#039;t be iterated over multiple times: the
   generator will need to be rebuilt by calling the generator function again.
  </p>

  <div class="simplesect">
   <h3 class="title">See Also</h3>
   <p class="para">
    <ul class="simplelist">
     <li><a href="language.oop5.iterations.php" class="link">Object Iteration</a></li>
    </ul>
   </p>
  </div>

 </div><?php manual_footer($setup); ?>