CSCI 136 :: Spring 2021
Data Structures & Advanced Programming
Home | Schedule | Labs | Handouts | Links | CS@Williams
Lab 7: The Two Towers
Your goal this week is to solve a difficult problem using Iterators. In fact, the problem is so difficult that there is no known efficient solution, so we will use "brute force". We will build an iterator to examine all possible configurations of objects from a set, keeping track of the best configuration that our iteration encounters. If we truly considered every possibility, then we must have found the best configuration! (Note: This lab is a variation of the Lab from Chapter 8, modified to use generic classes.)
Pre-lab Warmup
      Before lab, please do the following:
      Design an iterator that iterates over the characters of a String.  Repeated calls to the iterator’s next() method will return the first character of the string, then the second character, and so on.
      More specifically, you are to design and implement a class called CharacterIterator that has the following constructor and method signatures (you may provide additional helper methods and instance variables as you see fit):
          public class CharacterIterator extends AbstractIterator<Character> {
              public CharacterIterator(String str) { ... }
              public Character next() { ... }
              public boolean hasNext() { ... }
              public void reset() { ... }
              public Character get() { ... }
          }
     
        The Character class is a wrapper class for
	primitive char values so that we can
	use char values with the
	generic AbstractIterator class.  You
	use Character much like one
	uses Integer for int values.  The
	Java compiler will automatically
	convert char values to Character
	objects as necessary via a technique called “autoboxing.”  You
	may use the following main method to test your
	code.
    
            public static void main(String[] args) {
                CharacterIterator ci = new CharacterIterator("Hello world!");
                for (char c : ci) {
                    System.out.println(c);
                }
            }
       
    To refresh your memory, you may want to refer to Chapter 8 in your textbook to review Iterators. We recommend that you complete your Prelab Warmup solution before lab. You will submit it (in a file named CharacterIterator.java) in your git repository at the end of the lab.
Lab Assignment
Goal. To solve a computationally complex problem using iterators. Note: This lab is a variation of the one given in Chapter 8, modified to use generic classes.
Discussion. Suppose that we are given n uniquely-sized cubic blocks and that each block has a distinct face area between 1 and n. If we build two towers by stacking these blocks, how close can we make their heights? The following two towers, built by stacking 15 blocks, differ in height by only 129 millionths of an inch (each unit is one-tenth of an inch):
 
        Still, this stacking is only the second-best solution! To find the best stacking, we could consider all the possible configurations; if we truly consider every possible configuration, then if there is a better solution, we will find it.
We do know one thing: the total height of the two towers can be computed by summing the heights of all the blocks. Since the height of each block is the square root of its face area, this is:
If we consider all the subsets of the n blocks, we can think of each subset as representing the set of blocks that make up, say, the left tower (the "missing" blocks would then make up the right tower). We need only keep track of the subset that comes closest to h/2 without exceeding it (where h is the total height of all n blocks).
Note: In the figure above (taken from the Java Structures text), the stack on the left is actually the taller of the two!
      In this lab, we will represent a set of n distinct
      objects with a Vector<Double>, and we will
      construct an Iterator that returns each of
      the 2n subsets, one at a time.
    
Understanding the Algorithm
      The trick to understanding how to generate a subset
      of n values from a Vector is to first
      consider how to generate a subset of indices of elements
      from 0 to n-1. Once this simpler problem is
      solved, we can use the indices to help us build
      a Vector (or subset) of values identified by the
      indices.
    
      There are exactly 2n subsets of the values
      from 0 to n−1.  To see this, imagine that you
      have a strip of paper with n blanks on it,
      labeled n-1 to 0 (the blanks are labeled in
      descending order).  Now choose an arbitrary subset of blocks.  If a
      block i is included in a subset, you write
      a 1 in blank i on the paper, otherwise you
      put a 0 in blank i.  Since there are 2
      * 2 * ... * 2 = 2n different ways to fill in
      this strip of paper, there are 2n different
      subsets.
    
For example, suppose we start with a blank strip of paper.
 
  Also suppose that we have eight blocks, labeled 1,2,...,8, and we arbitrarily choose blocks 1, 4, 5, 6, and 7 as our subset:
 
  
      If we filled in the corresponding boxes in for each chosen block
      (i.e., if the block i is chosen, we put
      a 1 in spot i-1 since our blocks are
      1-indexed), our paper strip looks like this:
    
 
  
      Conveniently, we can think of the combination of 0s and 1s on
      strip of paper as the digits of an n-bit binary number.
      For example, the binary number 01111001 is the
      decimal number 121, since 01111001 is
      20 + 23 + 24 + 25 +
      26.
    
In fact, each number in the range 0 through 2n−1, when written in binary, uniquely describes a different subset. Given this, we develop a line of attack: count from 0 to 2n−1. For each number, use the value of the binary digits (a binary digit is called a bit) to select which blocks are to be included in a subset.
Working with Binary Numbers in Java
Computer scientists work with binary numbers frequently, so there are a number of useful things to remember:
int type is represented by 32 bits.  A
    Java long is represented by 64 bits.  For maximum
    flexibility, it is useful to use long values to
    represent sets of up to 64 elements.
  << can be
    used to quickly compute powers of 2.  The
    value 2i can be computed by shifting the
    binary value (1) i places to the left. In
    Java we write this 1L << i. This trick works
    only for non-negative, integral powers. The
    constant 1L is the value one stored as a
    64-bit long value. Using this constant ensures that
    we are using a 64-bit shift operation resulting in
    a long value instead of a 32-bit operation resulting
    in an int value. The L is important to
    ensure that the result is a long.
  long
    value m, we can use the experession m & (1L
    << i). If the expression yields the value 0, then the
    bit at position i is 0; if the expression yields the
    value (1L << i), then the bit at
    position i is 1.
  
Procedure
Armed with this information, the process of generating subsets is achievable. One approach is the following:
Implement an efficient iterator for subsets of a set.
AbstractIterator
	class.  This new class should have a constructor that takes
	a Vector<E> as its sole argument.  Subsets
	of this Vector will be returned as
	the Iterator progresses (e.g., each call
	to next() will create a new Vector object that is
	filled with values corresponding to the appropriate blocks).
	Name this extended class SubsetIterator, and be
	sure to import structure5.* at the top of your
	file.  Your SubsetIterator should be completely
	generic.  It should know nothing about the values it is
	iterating over. Thus, the declaration will be:
	 public class SubsetIterator<E> extends AbstractIterator<Vector<E>>SubsetIterator class is parameterized by the
	type <E>, and each step through the iteration
	yields a Vector<E>.
      long value is used to represent the
	current subset.  This value increases from 0 (the
	empty set) to 2n−1 (the entire set of
	values) as the Iterator progresses.
      reset method that resets the subset
	counter to 0.
      hasNext method that
	returns true if the iterator has not yet
	dispensed all possible subsets.
      get method that returns a
	new Vector<E> of values that are part of
	the current subset: if bit i of the current counter
	is 1, element i of the
	original Vector is included in the resulting
	subset Vector.
      next method. Remember it returns the
	current subset before incrementing the counter.
      Iterator you would normally have to write
	a remove method.  If you extend
	the AbstractIterator class, this method is
	provided and will do nothing (this is reasonable).
      
      Incrementally test.
      You can now test
      your SubsetIterator by asking it to print all the
      subsets of a Vector of values. For example, write
      a main method for your SubsetIterator
      class that creates a Vector<Integer> with the
      first 8 Integers (0 through 7),
      creates a SubsetIterator<Integer> with
      this Vector<Integer>, and then prints out all
      subsets returned. Make sure you end up with all 256
      different subsets printed.
    
Implement a Two Towers class.
To solve the two towers problem, write a main method in a new class called TwoTowers that takes an argument n from the command line.  For example,
        $ java TwoTowers 15
should compute the solution of the two towers problem for blocks labeled 1 through 15.
      It is easier to proceed by populating your Vector
      with height values instead of area values.  In other words,
      insert
       into a 
Vector<Double> object.
      To compute the square root of n, you can use
      the Math.sqrt(n) method.
      A SubsetIterator is then used to
      construct all 2n subsets of these values. The
      values of each subset are summed, and the sum that comes closest
      to, but does not exceed, the value h/2 is
      remembered. After all the subsets have been considered, print
      the best solution (the left and right towers). Since each block’s height is a square root, you
      should print out the area instead (e.g., print 2 instead of 1.41421356237).
In addition to printing the best solution, your program should also print the second best solution. In addition to providing an interesting twist to the problem, this has the effect of making it so that you can check the output of your program against the two towers diagram above.
The following is a sample run of the tool:
          $ java TwoTowers 14
          There are 14 total blocks.
          The half height (h/2) is: 18.298106626967595
          The best subset (left stack) is: [1, 4, 5, 8, 10, 12, 13] = 18.2964256530161
          The second best subset (left stack) is: [5, 8, 9, 10, 12, 13] = 18.2964256530161
Thought Questions
Be sure to answer the following questions in the PROBLEMS.md file, and submit it with the rest of your code this week.
time command, as in the following:
     $ time java -Xint TwoTowers 20
-Xint flag turns off some optimizations in the Java Virtual Machine and will give you more reliable results.)
    The output from the time program contains 3 lines:
     real 0m1.588s
 user 0m1.276s
 sys  0m0.922s
time manual page ($ man 1 time), you can see the meaning of each line. The real line shows the “wall clock” time, which is what we care about. Please run the program 3 times and report the median.
    randomSubset, a new SubsetIterator method that returns a random subset? Describe your strategy. You do not need to actually implement it. (Although if you have time, it's
    pretty straight-forward and you can get some interesing results!)
      checkstyle requirements:
    
      For this lab, we will be not be adding any
      new checkstyle rules to the set of rules that we
      have used in previous weeks.
    
We STRONGLY ENCOURAGE you to run checkstyle early and often when developing your code, and try to program in a way that minimizes WARNING messages. The checkstyle rules that we use in this course are based on real-world style guides; internalizing good style practices will help us write more readable code.
In total, checkstyle will enforce the following guidelines:
final must be
	declared private or protected (i.e.,
	no public member variables unless they are
	constants). (We don't expect this to be an issue this week.)
      public methods must include a “Javadoc” comment
	(starts with /** and ends with */;
	it should include descriptions of the function at the top,
	descriptions of return values after a @return tag,
	descriptions of each argument after a @param tag,
	and pre/post conditions after the @pre
	or @post tags).
      
      To run checkstyle, you would type the following
      command at the terminal:
    
        $ ./checkstyle
      The ./ is peculiar to Unix: it tells the terminal
      to look for the checkstyle program in the current
      directory.  This command will run checkstyle on
      every Java program in your directory.  To
      run checkstyle on a specific Java file, type:
    
        $ ./checkstyle SomeFile.java
Lab Deliverables
For this lab, please submit the following:
As in all labs, you will be graded on design, documentation, style, and correctness. Be sure to document your program with appropriate comments, including a general description at the top of each Java class, a description of each method with pre- and post-conditions where appropriate. Also use comments and descriptive variable names to clarify sections of the code which may not be clear to someone trying to understand it.
Submitting Your Lab
As you complete various milestones, you should commit your changes and push them. Commit early and often. When the deadline arrives, we will retrieve the latest version of your code. If you are confident that you are done, please include "Lab Submission" as the commit message for your final commit. If you later decide that you have more edits to make, it is OK. We will look at the latest commit before the deadline.
We will know that the files are yours because they are in your git repository. Do not include identifying information in the code that you submit. Our goal is to grade the programs anonymously to avoid any bias. However, in your README.md file, please cite any sources of inspiration or collaboration (e.g., conversations with classmates). We take the honor code very seriously, and so should you. Please include the statement "I am the sole author of the work in this repository" in the comments at the top each of your java files.