CSE 326 Data Structures
Project 1

Code due: Wednesday, January 20, 1999, electronically by 11:30 pm
Documentation due: Thursday, January 21, 1999, in section

Objectives

The purpose of this assignment is to:

Overview

Stacks and Queues are two very basic abstract data types for storing data that are useful in a wide variety of applications. Stacks have a Last-In-First-Out (LIFO) behavior: if a number of items are put on a stack, they will be removed from the stack in reverse order. Queues have a First-In-First-Out (FIFO) behavior: items leave a queue in the order that they were placed on the queue.

Despite their different behavior, one can implement a queue data structure by employing two instances of a stack data structure. This implementation is efficient: n queue operations can be performed using on average a constant number of stack operations per queue operation. Assuming that each stack operation takes a constant amount of work, then performing n queue operations can take only linear (O(n)) time.

Assignment

Your job is to implement a queue data structure that stores its data using only two instances of a stack data structure. Your implementation must be efficient: you will be required to argue that it uses only O(n) stack operations for n queue operations.

What we've provided

As a starting point we've provided a few files:

All provided files can be found in the project directory:
/cse/courses/cse326/1999wi/project1
You should make a copy of this directory into your home directory
% cp -r /cse/courses/cse326/1999wi/project1 ~/project1
and edit your copy.

Handing in the code

You are required to hand in your project1 directory with your implementation of the Queue data structure. Included in this directory should one file Queue.h. This should contain your implementation of the Queue data structure that uses two Stacks. Your implementation should contain sufficient comments so that together with the written documentation a skilled programmer can understand your program. You should strive for simple, elegant, and efficient solutions. Be willing to sacrifice some efficiency (speed) for simplicity and elegance. We are using the turnin program to hand in project assignments. When you are ready to submit your solution, you should have all of your files sitting in some directory. Suppose this directory is named ~/326stuff/myproj1dir. Suppose also, that you're in section AA. To submit your solution, type the following:

% cd ~/326stuff/myproj1dir
% make clean
% cd ../
% turnin -c cse326=AA -p project1 myproj1dir
This will submit the contents of your directory with a time stamp. You can submit more than once, but only the last submission is kept. Check the contents of your last submission by:
% turnin -v -c cse326=AA -p project1
man turnin for more details and options.

Handing in the documentation

Your documentation should be about one or two type written pages. Your figures can be hand drawn. It should include the following:

  1. Describe your queue implementation. Specifically, how did you implement the Enqueue() and Dequeue() operations? Provide interesting examples illustrating the queue's behavior.
  2. Argue that your implementation is efficient. First, what is the worst case running time of your queue for a single Enqueue() operation? Dequeue() operation? Despite this worst case behavior, why does your implementation use O(n) time for n queue operations? Assume that stack operations take a constant amount of time.
Your writing style is important on this part of the project. Your documentation should be well organized and clear. There should be a title and section headings if needed. You should write to an audience of moderately skilled programmers and computer scientists. The reader should come away from your documentation with a clear understanding of your solution and its properties. The documentation is submitted in class.

Computing at home

It is fine to develop your program at home using another C++ compiler, but you must turn in the program using turnin and your program must compile and work correctly using g++ on the instructional machines. In this way we can compile and run your program on our test suite.

C++ Templates

Many ADTs that we will describe in this class are general enough to manage many types of data. For example, this assignment uses a queue of strings, but you could imagine having queues of integers, or queues of points, etc. Unfortunately, C++ does not make it easy to implement ADTs for general purpose use. One way to do this in C++ is to use templates.

Look at our Stack implementation using templates in Stack.h. It illustrates the general structure for declaring a template class:

template <class T> class Stack {
...
    // Stack specification and implementation with respect to type T
...
};
Everything inside the brackets is just like it would be in normal class specification/implementations, except the declarations use some unknown type T. The identifier T is used as a placeholder to represent the type of data that will be stored on the stack.

Instantiating template classes to create new objects is similar to normal classes, except we must specify the type for the template. For example, here is how you create a stack of integers capable of holding 100 objects:

Stack<int> mystack(100);

mystack.Push(8);
mystack.Push(4);
int result = mystack.Pop();
Note that its use after instantiation is just like any other object.

The only caveat of using C++ templates is that both the specification and implementation must be specified in an include file. This is because the compiler cannot compile code for a template class until it sees how it is being instantiated. If code needs to have both int stacks (Stack<int>) and (char *) stacks (Stack<char *>), the compiler will generate different code for each. The end result, however, is what we desired: the ability to specify both types of stacks with a single set of source code.

Note that the recent C++ standard allows for compilers to support template class implementation in a separate source module. However, most compilers including our version of g++ on the instructional unix machines don't support this.