Download Presentation
## Stacks

- - - - - - - - - - - - - - - - - - - - - - - - - - - E N D - - - - - - - - - - - - - - - - - - - - - - - - - - -

**Outline**3.2 This topic discusses the concept of a stack: • Description of an Abstract Stack • List applications • Implementation • Example applications • Parsing: XHTML, C++ • Function calls • Reverse-Polish calculators • Robert’s Rules • Standard Template Library**Abstract Stack**3.2.1 An Abstract Stack (Stack ADT) is an abstract data type which emphasizes specific operations: • Uses a explicit linear ordering • Insertions and removals are performed individually • Inserted objects are pushed onto the stack • The top of the stack is the most recently object pushed onto the stack • When an object is popped from the stack, the current top is erased**Abstract Stack**3.2.1 Also called a last-in–first-out (LIFO) behaviour • Graphically, we may view these operations as follows: There are two exceptions associated with abstract stacks: • It is an undefined operation to call either pop or top on an empty stack**Applications**3.2.2 Numerous applications: • Parsing code: • Matching parenthesis • XML (e.g., XHTML) • Tracking function calls • Dealing with undo/redo operations • Reverse-Polish calculators • Assembly language The stack is a very simple data structure • Given any problem, if it is possible to use a stack, this significantly simplifies the solution**Stack: Applications**3.2.2 Problem solving: • Solving one problem may lead to subsequent problems • These problems may result in further problems • As problems are solved, your focus shifts back to the problem which lead to the solved problem Notice that function calls behave similarly: • A function is a collection of code which solves a problem Reference: Donald Knuth**Implementations**3.2.3 We will look at two implementations of stacks: The optimal asymptotic run time of any algorithm is Q(1) • The run time of the algorithm is independent of the number of objects being stored in the container • We will always attempt to achieve this lower bound We will look at • Singly linked lists • One-ended arrays**Linked-List Implementation**3.2.3.1 Operations at the front of a singly linked list are all Q(1) The desired behaviour of an Abstract Stack may be reproduced by performing all operations at the front**Single_list Definition**3.2.3.1 The definition of single list class from Project 1 is: template <typename Type> class Single_list { public: Single_list(); ~Single_list(); int size() const; bool empty() const; Type front() const; Type back() const; Single_node<Type> *head() const; Single_node<Type> *tail() const; int count( Type const & ) const; void push_front( Type const & ); void push_back( Type const & ); Type pop_front(); int erase( Type const & ); };**Stack-as-List Class**3.2.3.1 The stack class using a singly linked list has a single private member variable: template <typename Type> class Stack { private: Single_list<Type> list; public: boolempty() const; Typetop() const; voidpush( Type const& ); Typepop(); };**Stack-as-List Class**A constructor and destructor is not needed Because list is declared, the compiler will call the constructor of the Single_list class when the Stack is constructed 3.2.3.1 template <typename Type> class Stack { private: Single_list<Type> list; public: boolempty() const; Typetop() const; voidpush( Type const& ); Typepop(); };**Stack-as-List Class**The empty and push functions just call the appropriate functions of the Single_list class template <typename Type> bool Stack<Type>::empty() const { return list.empty(); } template <typename Type> void Stack<Type>::push( Type const &obj ) { list.push_front( obj ); } 3.2.3.1**Stack-as-List Class**3.2.3.1 The top and pop functions, however, must check the boundary case: template <typename Type> Type Stack<Type>::top() const { if ( empty() ) { throw underflow(); } return list.front(); } template <typename Type> Type Stack<Type>::pop() { if ( empty() ) { throw underflow(); } return list.pop_front(); }**Array Implementation**3.2.3.2 For one-ended arrays, all operations at the back are Q(1)**Destructor**We need to store an array: In C++, this is done by storing the address of the first entry Type *array; We need additional information, including: The number of objects currently in the stack int stack_size; The capacity of the array int array_capacity; 3.2.3.2**Stack-as-Array Class**3.2.3.2 We need to store an array: • In C++, this is done by storing the address of the first entry template <typename Type> class Stack { private: intstack_size; intarray_capacity; Type *array; public: Stack( int = 10 ); ~Stack(); boolempty() const; Typetop() const; voidpush( Type const & ); Typepop(); };**Constructor**3.2.3.2 The class is only storing the address of the array • We must allocate memory for the array and initialize the member variables • The call to new Type[array_capacity] makes a request to the operating system for array_capacity objects #include <algorithm> // ... template <typename Type> Stack<Type>::Stack( intn ): stack_size( 0 ), array_capacity( std::max( 1, n ) ), array( new Type[array_capacity] ) { // Empty constructor }**Constructor**Warning: in C++, the variables are initialized in the order in which they are defined: 3.2.3.2 template <typename Type> class Stack { private: intstack_size; intarray_capacity; Type *array; public: Stack( int = 10 ); ~Stack(); boolempty() const; Typetop() const; voidpush( Type const& ); Typepop(); }; template <typename Type> Stack<Type>::Stack( intn ): stack_size( 0 ), array_capacity( std::max( 1, n ) ), array( new Type[array_capacity] ) { // Empty constructor }**Destructor**3.2.3.2 The call to new in the constructor requested memory from the operating system • The destructor must return that memory to the operating system: template <typename Type> Stack<Type>::~Stack() { delete [] array; }**Empty**3.2.3.2 The stack is empty if the stack size is zero: template <typename Type> bool Stack<Type>::empty() const { return ( stack_size == 0 ); } The following is unnecessarily tedious: • The == operator evaluates to either true or false if ( stack_size == 0 ) { return true; } else { return false; }**Top**If there are n objects in the stack, the last is located at index n – 1 template <typename Type> Type Stack<Type>::top() const { if ( empty() ) { throw underflow(); } return array[stack_size - 1]; } 3.2.3.2**Pop**Removing an object simply involves reducing the size It is invalid to assign the last entry to “0” By decreasing the size, the previous top of the stack is now at the location stack_size template <typename Type> Type Stack<Type>::pop() { if ( empty() ) { throw underflow(); } --stack_size; return array[stack_size]; } 3.2.3.2**Push**3.2.3.2 Pushing an object onto the stack can only be performed if thearray is not full template <typename Type> void Stack<Type>::push( Type const &obj ) { if ( stack_size == array_capacity ) { throw overflow(); // Best solution????? } array[stack_size] = obj; ++stack_size; }**Exceptions**The case where the array is full is not an exception defined in the Abstract Stack If the array is filled, we have five options: Increase the size of the array Throw an exception Ignore the element being pushed Replace the current top of the stack Put the pushing process to “sleep” until something else removesthe top of the stack Include a member function bool full() const; 3.2.3.2**Array Capacity**If dynamic memory is available, the best option is to increase the array capacity If we increase the array capacity, the question is: How much? By a constant? array_capacity += c; By a multiple? array_capacity *= c; 3.2.4**Array Capacity**First, let us visualize what must occur to allocate new memory 3.2.4**Array Capacity**First, this requires a call to new Type[N] where N is the new capacity We must have access to this so we muststore the address returned by new in alocal variable, say tmp 3.2.4**Array Capacity**Next, the values must be copied over 3.2.4**Array Capacity**The memory for the original array must be deallocated 3.2.4 W**Array Capacity**Finally, the appropriate member variables must be reassigned 3.2.4**Array Capacity**The implementation: void double_capacity() { Type *tmp_array = new Type[2*array_capacity]; } 3.2.4**Array Capacity**The implementation: void double_capacity() { Type *tmp_array = new Type[2*array_capacity]; } 3.2.4 tmp_array**Array Capacity**The implementation: void double_capacity() { Type *tmp_array = new Type[2*array_capacity]; for ( int i = 0; i < array_capacity; ++i ) { tmp_array[i] = array[i]; } } 3.2.4 tmp_array**Array Capacity**The implementation: void double_capacity() { Type *tmp_array = new Type[2*array_capacity]; for ( int i = 0; i < array_capacity; ++i ) { tmp_array[i] = array[i]; } delete [] array; } 3.2.4 tmp_array W**Array Capacity**The implementation: void double_capacity() { Type *tmp_array = new Type[2*array_capacity]; for ( int i = 0; i < array_capacity; ++i ) { tmp_array[i] = array[i]; } delete [] array; array = tmp_array; array_capacity *= 2; } 3.2.4 tmp_array**Array Capacity**Back to the original question: How much do we change the capacity? Add a constant? Multiply by a constant? First, we recognize that any time that we push onto a full stack, this requires n copies and the run time is Q(n) Therefore, push is usually Q(1) except when new memory is required 3.2.4**Array Capacity**To state the average run time, we will introduce the concept of amortized time: If n operations requires Q(f(n)), we will say that an individual operation has an amortized run time of Q(f(n)/n) Therefore, if inserting n objects requires: Q(n2) copies, the amortized time is Q(n) Q(n) copies, the amortized time is Q(1) 3.2.4**Array Capacity**Let us consider the case of increasing the capacity by 1 each time the array is full With each insertion when the array is full, this requires all entries to be copied 3.2.4**Array Capacity**Suppose we double the number of entries each timethe array is full Now the number of copies appears to be significantlyfewer 3.2.4**Array Capacity**Suppose we insert k objects The pushing of the kth object on the stack requires k copies The total number of copies is now given by: Therefore, the amortized number of copiesis given by Therefore each push must run inQ(n) time The wasted space, howeveris Q(1) 3.2.4.1**Array Capacity**Suppose we double the array size each time it is full: We must make 1, 2, 4, 8, 16, 32, 64, 128, ... copies Inserting n objects would therefore require 1, 2, 4, 8, ..., all theway up to the largest 2k < n or Therefore the amortized number ofcopies per insertion is Q(1) The wasted space,however is O(n) 3.2.4.2**Array Capacity**What if we increase the array size by a larger constant? For example, increase the array size by 4, 8, 100? 3.2.4.3**Array Capacity**Here we view the number of copies required when increasing the array size by 4; however, in general, suppose we increase it by a constant value m Therefore, the amortized run time perinsertion is Q(n) 3.2.4.3**Array Capacity**Note the difference in worst-case amortized scenarios: The web site http://www.ece.uwaterloo.ca/~ece250/Algorithms/Array_resizing/ discusses the consequences of various values of r 3.2.4.3**Application: Parsing**3.2.5 Most parsing uses stacks Examples includes: • Matching tags in XHTML • In C++, matching • parentheses ( ... ) • brackets, and [ ... ] • braces { ... }**Parsing XHTML**3.2.5.1 The first example will demonstrate parsing XHTML We will show how stacks may be used to parse an XHTML document You will use XHTML (and more generally XML and other markup languages) in the workplace**Parsing XHTML**3.2.5.1 A markup language is a means of annotating a document to given context to the text • The annotations give information about the structure or presentation of the text The best known example is HTML, or HyperText Markup Language • We will look at XHTML**Parsing XHTML**3.2.5.1 XHTML is made of nested • opening tags, e.g., <some_identifier>, and • matching closing tags, e.g., </some_identifier> <html> <head><title>Hello</title></head> <body><p>This appears in the <i>browswer</i>.</p></body> </html>**Parsing XHTML**3.2.5.1 Nesting indicates that any closing tag must match the most recent opening tag Strategy for parsing XHTML: • read though the XHTML linearly • place the opening tags in a stack • when a closing tag is encountered, check that it matches what is on top of the stack and**Parsing XHTML**3.2.5.1 <html> <head><title>Hello</title></head> <body><p>This appears in the <i>browswer</i>.</p></body> </html>