Close
Register
Close Window

Masters of Engineering Bridge Course

Chapter 11 Lists and Iterators

Show Source |    | About   «  11.2. Unit 11 Lab 1 Placeholder   ::   Contents   ::   11.4. Unit 11 Lab 2 Placeholder  »

11.3. Iterators

11.3.1. Objectives

Upon completion of this module, students will be able to:

  • Describe the purpose and use of an Iterator

  • Implement Iterators using the Iterator and Iterable Interfaces

  • Design and develop algorithms that use Iterators and Iterator methods

11.3.2. Introduction to Iterators

11.3.2.1. [13:14] Introduction to Iterators Video

Video Slides 11.3.2.1-IntroToIterators.pdf

11.3.3. Checkpoint 1

11.3.4. Programming Using the Iterable Interface

11.3.4.1. [4:36] Programming Using the Iterable Interface Video

Video Slides 11.3.3.1-Iterable.pdf

11.3.4.2. Checkpoint 2

11.3.5. Programming Using Iterators

11.3.5.1. [18:02] Programming Using Iterators Video

Video Slides 11.3.4.1-ProgrammingWithIterators.pdf

11.3.5.2. Checkpoint 3

11.3.6. Iterator Design Decisions

11.3.6.1. [8:21] Iterator Design Decisions Video

Video Slides11.3.5.1-IteratorsDesignConsiderations.pdf

Clarification: Iterators that are nested class inside the linked structure (not subclasses) are more efficient than Iterators that are independent classes.

11.3.7. Inner Iterator for 11.3.4.1-IteratorExample

As discussed throughout this section there are various design approaches for iterators. Below is one example of how an inner Iterator class could be implemented for 11.3.4.1-IteratorExample.

Include a public method to make the iterator object available:

/**
* Iterator method creates Iterator object
*
* @return new Iterator object
*/
public Iterator<T> iterator()
{
   return new LListIterator<T>();
}

Include an inner Iterator class. This version does not provide remove functionality as it is complicated with a singly linked list to keep track of the previous nodes in order to remove the current node.

private class LListIterator<A> implements Iterator<T>
{
     private Node next;
     private boolean newCurr;

     /**
     * Creates a new DLListIterator
     */
     public LListIterator()
     {
       next = firstNode;
       newCurr = false;
     }

     /**
     * Checks if there are more elements in the list
     *
     * @return true if there are more elements in the list
     */
     @Override
     public boolean hasNext()
     {
       return (next != null);
     }

     /**
     * Gets the next value in the list
     *
     * @return the next value
     * @throws NoSuchElementException
     *             if there are no nodes left in the list
     */
     @Override
     public T next()
     {
       if (next == null)
       {
         throw new NoSuchElementException("No nodes left in the list.");
       }
       T value = next.data;
       next = next.getNext();
       newCurr = true;
       return value;
     }
}

A version of an inner Iterator class which does provide remove functionality. It is best to only provide remove functionality through either the data structure or the iterator in order to avoid unintended side effects.

private class LListIterator<A> implements Iterator<T>
 {
     private Node prev;
     private Node curr;
     private Node next;
     private boolean newCurr;

     /**
     * Creates a new DLListIterator
     */
     public LListIterator()
     {
         prev = null;
         curr = null;
         next = firstNode;
         newCurr = false;
     }

     /**
     * Checks if there are more elements in the list
     *
     * @return true if there are more elements in the list
     */
     @Override
     public boolean hasNext()
     {
         return (next != null);
     }

     /**
     * Gets the next value in the list
     *
     * @return the next value
     * @throws NoSuchElementException
     *             if there are no nodes left in the list
     */
     @Override
     public T next()
     {
         prev = curr;
         curr = next;
         next = next.getNext();
         if (curr == null)
         {
             throw new NoSuchElementException("No nodes left in the list.");
         }
         newCurr = true;
         return curr.data;
     }

    /**
     * Removes the last object returned with next() from the list
     *
     * @throws IllegalStateException
     *             if next has not been called yet
     *             and if the element has already been removed
     */
     @Override
     public void remove()
     {
         if (next == firstNode)
         {
             throw new IllegalStateException(
                  "Next has not been called yet.");
         }
         else if (!newCurr)
         {
             throw new IllegalStateException(
                  "The Element has already been removed.");
         }
         else if (curr == firstNode) {
             firstNode = next;
             curr = null;
         } else {
             prev.setNext(curr.getNext());
             curr = prev;
              //this code that updates prev is not necessary
              //because next() must be called before another remove()
              //and that will update prev, saving this O(n) operation
              //prev = firstNode;
              //while ((prev != null) && (prev.getNext() != curr)){
              //    prev = prev.getNext();
              //}
         }
         numberOfEntries--;
         newCurr = false;
     }
 }

11.3.8. Programming Practice: Iterators

   «  11.2. Unit 11 Lab 1 Placeholder   ::   Contents   ::   11.4. Unit 11 Lab 2 Placeholder  »

nsf
Close Window