940 likes | 1.12k Views
Chapter 5. Topics. Stacks Queues Deque Adapters and the Adapter Design Pattern. Stacks. Stacks. A data structure where addition and deletion of elements take place only at one end, called the top of the stack A last in first out (LIFO) data structure
E N D
Topics • Stacks • Queues • Deque • Adapters and the Adapter Design Pattern
Stacks • A data structure where addition and deletion of elements take place only at one end, called the top of the stack • A last in first out (LIFO) data structure • In a cafeteria the second tray, in a stack of trays, can be removed only if the first tray is removed
Stacks Applications of Stacks • Direct applications • Page-visited history in a Web browser • Undo sequence in a text editor • Chain of method calls in the C++ run-time system • Indirect applications • Auxiliary data structure for algorithms • Component of other data structures
An abstract data type (ADT) is an abstraction of a data structure An ADT specifies: Data stored Operations on the data Error conditions associated with operations Example: ADT modeling a simple stock trading system The data stored are buy/sell orders The operations supported are order buy(stock, shares, price) order sell(stock, shares, price) void cancel(order) Error conditions: Buy/sell a nonexistent stock Cancel a nonexistent order Abstract Data Types (ADTs)
The Stack ADT stores arbitrary objects Insertions and deletions follow the last-in first-out scheme Think of a spring-loaded plate dispenser Main stack operations: push(object): inserts an element object pop(): removes and returns the last inserted element Auxiliary stack operations: object top(): returns the last inserted element without removing it integer size(): returns the number of elements stored boolean isEmpty(): indicates whether no elements are stored The Stack ADT PEZ dispenser
C++ Run-Time Stack main() { int i = 5; foo(i); } foo(int j) { int k; k = j+1; bar(k); } bar(int m) { … } • The C++ run-time system keeps track of the chain of active functions with a stack • When a function is called, the system pushes on the stack a frame containing • Local variables and return value • Program counter, keeping track of the statement being executed • When the function ends, its frame is popped from the stack and control is passed to the function on top of the stack • Allows for recursion bar PC = 1 m = 6 foo PC = 3 j = 5 k = 6 main PC = 2 i = 5
Containers • Container • An object that can hold other objects as its elements • Goal: • Create a container object that automatically grows or shrinks as needed • One way to accomplish this is to use a STL container class that manages a sequence of elements • Memory is allocated and de-allocated as needed
More on Containers • Problem • Want to read in an unknown number of values into a buffer • Size of the buffer is unknown • One method to solve problem is to use a STL container class called a vector • A vector is similar to an array except that can automatically expand and contract as needed • The STL containers have an embedded allocator that manages memory • The new and delete operators are not used by the programmer
Still more on Containers • Can store objects of any type • Specify type by using < > • C++ template • Need to include the required header
STL Template Headers • <vector> - defines a template class for implementing a container (resizable array) • <stack> - container with the last-in, first-out access • <queue> - container with the first-in, first-out access • <deque> - double ended queue • <list> - doubly linked list • <priority_queue> - queue order by value • <set> - set • <map> - associative array (dictionary) • <iterator> - defines a template for defining manipulating iterators
STL iterators • iterator • Generalized pointer for keeping track of the beginning, ending, and other positions of a data structure • Is a class object that can point at an element of a container by storing its memory address • Has built in operations that can access the value stored at a location • iterator++ // move to the next element • iterator-- // move to the previous element • *iterator // access the value at the position pointed to by iterator
STL constructors • Constructor syntax • vector < type> v //construct v as a vector <type> of capacity 0 • vector < type> v(n) //construct v as a vector <type> of capacity n, size n, and each element is initialized to the default type value • vector < type> v(n, initialValue) // construct v as a vector <type> of capacity n, size n, and each element is initialized to initialValue
STL Member Functions (1) • Given vector <type> v • v.size( ) //returns the number of values v currently contains • v.empty( ) is a faster alternative to the boolean expression v.size ( ) == 0 • v.capacity ( ) returns the current capacity of v • v.push_back(value) // append value at v’s end • v.reserve (n) grows v so its capacity in n (does not affect v’s size) • v.pop_back( ) // erase v’s last element
STL Member Functions (2) • v.front( ) // returns a reference to v’s first element • v.back( ) // returns a reference to v’s last element • v.begin( ) // returns a iterator positioned at v’s first value • v.end ( ) // returns an iterator positioned immediately after v’s last value • v.insert(pos, value ) // inserts value into v at iterator position pos • v.erase(pos) // erases the value in v at iterator position pos • v.erase(pos1, pos2) // erase the values in v from iterator position pos1 to pos2
STL Member Functions (3) • find(begin,end,value) // returns an iterator to value in the unsorted sequence, if not present return end • sort(begin,end) // sorts the sequence in ascending order • random_shuffle(begin,end) // shuffles the values in the sequence randomly • The subscript operator, [ ], does not update v’s size or capacity • The push_back ( ) method should be used when adding values to v • The push_back ( ) method updates the iterator returned by the end ( ) method • The subscript operator should only be used to access or change a value (see examples STL1.cpp through STL6.CPP)
The STL Stack • STL provides an implementation of a stack • Based on the STL vector class • Space is dynamically managed by STL • Need to #include <stack> • Part of the namepsace std • To create a stack of doubles: • stack <double> myStack
The STL Stack Methods • size() : • Returns the number elements in the stack • empty() : • Returns true if the stack is empty else false • push(e) : • Push e onto the top of the stack • pop() : • Pop the element at the top of the stack • top() : • Returns a reference to the element at the top of the stack • An exception is not thrown resulting from a pop() or top() to an empty stack • Programmer’s responsibility • (see STL7.cpp)
Stack Interface in C++ template <typename E> class Stack { public: int size() const; bool empty() const; const E& top() const throw(StackEmpty); void push(const E& e); void pop() throw(StackEmpty); } Code Fragment 5.1 • C++ interface corresponding to a Stack ADT • Uses an exception class StackEmpty • Different from the built-in C++ STL class stack • (Implementing a Stack from “Scratch”) What does const mean in the above specifications? How does one save the “popped” value?
Attempting the execution of an operation of ADT may sometimes cause an error condition, called an exception Exceptions are said to be “thrown” by an operation that cannot be executed In the Stack ADT, operations pop and top cannot be performed if the stack is empty Attempting pop or top on an empty stack throws a StackEmptyexception Exceptions
Empty Stack Exception // Exception thrown on performing top or pop of an empty stack class StackEmpty : public RuntimeException { public: StackEmpty(const string& err) : RuntimeException(err) { } }; Code Fragment 5.2
Top element is stored in S(t) A simple way of implementing the Stack ADT uses an array One adds elements from left to right A variable keeps track of the index of the top element Array-Based Stack Algorithmsize() returnt + 1 Algorithmpop() ifempty()then throw StackEmpty else t t 1 returnS[t + 1] … S 0 1 2 t N - 1
… S 0 1 2 t Array-based Stack (cont.) • The array storing the stack elements may become full • A push operation will then throw a StackFull exception • Limitation of the array-based implementation • Not intrinsic to the Stack ADT Algorithmpush(o) ift= S.size() 1 then throw StackFull else t t + 1 S[t] o
Other Methods: Array-based Stack AlgorithmisEmpty() return (t < 0) Algorithmtop() if isEmpty then throw EmptyStackException return S[t]
Performance and Limitations • Performance • Let n be the number of elements in the stack • The space used is O(n) • Each operation runs in time O(1) • Limitations • The maximum size of the stack must be defined a priori and cannot be changed • Trying to push a new element into a full stack causes an implementation-specific exception
A C++ Implementation template <typename E> class ArrayStack { enum { DEF_CAPACITY = 100 }; // default stack capacity public: ArrayStack(int cap = DEF_CAPACITY); // constructor from capacity int size() const; // number of items in the stack bool empty() const; // is the stack empty? const E& top() const throw(StackEmpty); // get the top element void push(const E& e) throw(StackFull); // push element onto stack void pop() throw(StackEmpty); // pop the stack // ...housekeeping functions omitted private: // member data E* S; // array of stack elements int capacity; // stack capacity int t; // index of the top of the stack };
C++ Implementation Methods template <typename E> ArrayStack<E>::ArrayStack(int cap) : S(new E[cap]), capacity(cap), t(-1) { } // constructor from capacity template <typename E> intArrayStack<E>::size() const { return (t + 1); } // number of items in the stack template <typename E> boolArrayStack<E>::empty() const { return (t < 0); } // is the stack empty? template <typename E> // return top of stack const E& ArrayStack<E>::top() const throw(StackEmpty) { if (empty()) throw StackEmpty("Top of empty stack"); return S[t]; }
C++ Implementation Methods - 2 template <typename E> // push element onto the stack void ArrayStack<E>::push(const E& e) throw(StackFull) { if (size() == capacity) throw StackFull("Push to full stack"); S[++t] = e; } template <typename E> // pop the stack void ArrayStack<E>::pop() throw(StackEmpty) { if (empty()) throw StackEmpty("Pop from empty stack"); --t; }
C++ Implementation Trace ArrayStack<int> A; // A = [], size = 0 A.push(7); // A = [7*], size = 1 A.push(13); // A = [7, 13*], size = 2 cout << A.top() << endl; A.pop(); // A = [7*], outputs: 13 A.push(9); // A = [7, 9*], size = 2 cout << A.top() << endl; // A = [7, 9*], outputs: 9 cout << A.top() << endl; A.pop(); // A = [7*], outputs: 9 ArrayStack<string> B(10); // B = [], size = 0 B.push("Bob"); // B = [Bob*], size = 1 B.push("Alice"); // B = [Bob, Alice*], size = 2 cout << B.top() << endl; B.pop(); // B = [Bob*], outputs: Alice B.push("Eve"); // B = [Bob, Eve*], size = 2 (See another array implementation)
Array-based Stack Drawbacks • A fixed upper bound must be assumed • Can waste memory or • Not enough memory allocated
Stack with a Singly Linked List • One can implement a stack with a singly linked list • The top element is stored at the first node of the list • It takes O(n) time to allocate space • Each operation of the Stack ADT takes O(1) time nodes t elements
Linked Implementation of Stacks • Because an array size is fixed, in the array (linear) representation of a stack, only a fixed number of elements can be pushed onto the stack • If in a program the number of elements to be pushed exceeds the size of the array, the program may terminate in an error • In a linked representation top is used to locate the top element in the stack • When using arrays, top gives the index of the array
Implementing a Stack with a Generic Linked List (see 5.7 – 5.9)
Reversing an Array template <typename E> void reverse(vector<E>& V) { // reverse a vector ArrayStack<E> S(V.size()); for (int i = 0; i < V.size(); i++) // push elements onto stack S.push(V[i]); for (int i = 0; i < V.size(); i++) { // pop them in reverse order V[i] = S.top(); S.pop(); } }
An Application of Stacks - Parentheses Matching • Each “(”, “{”, or “[” must be paired with a matching “)”, “}”, or “[” • correct: ( )(( )){([( )])} • correct: ((( )(( )){([( )])} • incorrect: )(( )){([( )])} • incorrect: ({[ ])} • incorrect: (
An Application of Stacks - Parentheses Matching Algorithm • Each time an opening symbol is encountered, the symbol is pushed on the stack • Each time a closing symbol is encountered, the symbol is popped off the stack (assuming it is not empty) • Ensure the symbols are of the corresponding type • If the stack is empty after the sequence is processed, then all the symbols match
Parentheses Matching Algorithm Algorithm ParenMatch(X,n): Input: An array X of n tokens, each of which is either a grouping symbol, a variable, an arithmetic operator, or a number Output: true if and only if all the grouping symbols in X match Let S be an empty stack for i=0 to n-1 do if X[i] is an opening grouping symbol then S.push(X[i]) else if X[i] is a closing grouping symbol then if S.isEmpty() then return false {nothing to match with} if S.pop() does not match the type of X[i] then return false {wrong type} if S.isEmpty() then return true {every symbol matched} else return false {some symbols were never matched}
<body> <center > <h1> The Little Boat </h1> </center> <p> The storm tossed the little boat like a cheap sneaker in an old washing machine. The three drunken fishermen were used to such treatment, of course, but not the tree salesman, who even as a stowaway now felt that he had overpaid for the voyage. </p> <ol> <li> Will the salesman die? </li> <li> What color is the boat? </li> <li> And what about Naomi? </li> </ol> </body> The Little Boat The storm tossed the little boat like a cheap sneaker in an old washing machine. The three drunken fishermen were used to such treatment, of course, but not the tree salesman, who even as a stowaway now felt that he had overpaid for the voyage. 1. Will the salesman die? 2. What color is the boat? 3. And what about Naomi? HTML Tag Matching For fully-correct HTML, each <name> should pair with a matching </name> (See 5.12 – 5.14)
Another Approach: Definition of Linked Nodes template <class Type> structnodeType { Type info; nodeType<Type> *link; };
Definition of Linked List template<class Type> class linkedStackType { public: boolisEmptyStack(); boolisFullStack(); void push(const Type& newItem); void pop(Type& poppedElement); void destroyStack(); linkedStackType(); linkedStackType(const linkedStackType<Type>& otherStack); ~linkedStackType(); private: nodeType<Type> *top; //pointer to the stack };
Pop Function template<class Type> void linkedStackType<Type>::pop(Type& poppedElement) { nodeType<Type> *temp; //pointer to deallocate memory poppedElement = top->info; //copy the top element temp = top; //set temp to point to the top node top = top->link; //advance top to the next node delete temp; //delete the top node }//end pop (See example linkstack.cpp)
Queues • A queue is a data structure in which elements are added at one end, called the rear, and deleted from the other end, called the front • A first in first out (FIFO) data structure • The middle elements of the queue are inaccessible • The rear of the queue is accessed whenever a new element is added to the queue • The front of the queue is accessed whenever an element is deleted from the queue • Consider a line of customers in a bank, where customers are waiting either to withdraw/deposit money or for some other business • Each new customer gets in the line at the rear and whenever a teller is ready for a new customer, the customer at the front of the line is served
Applications of Queues • Direct applications • Waiting lists, bureaucracy • Access to shared resources (e.g., printer) • Multiprogramming • Indirect applications • Auxiliary data structure for algorithms • Component of other data structures
The Queue ADT stores arbitrary objects Insertions and deletions follow the first-in first-out scheme Insertions are at the rear of the queue and removals are at the front of the queue Main queue operations: enqueue(object): inserts an element at the end of the queue object dequeue(): removes and returns the element at the front of the queue Auxiliary queue operations: object front(): returns the element at the front without removing it integer size(): returns the number of elements stored boolean isEmpty(): indicates whether no elements are stored Exceptions Attempting the execution of dequeue or front on an empty queue throws an EmptyQueueException The Queue ADT
Queue Interface in C++ • C++ interface corresponding to the Queue ADT • Requires the def-inition of exception QueueEmpty • No corresponding built-in C++ class template <typename E> class Queue { public: int size() const; bool empty() const; const E& front() const throw(QueueEmpty); void enqueue (const E& e); void dequeue() throw(QueueEmpty);}; 5.15
Empty Queue Exception // class QueueEmpty : public RuntimeException { public: QueueEmpty(const string& err) : RuntimeException(err) { } }; Code Fragment 5.16
Queue Example OperationOutputfront Q rear enqueue(5) – (5) enqueue(3) – (5, 3) dequeue() 5 (3) enqueue(7) – (3, 7) dequeue() 3 (7) front() 7 (7) dequeue() 7 () dequeue() “error” () isEmpty() true () enqueue(9) – (9) enqueue(7) – (9, 7) size() 2 (9, 7) enqueue(3) – (9, 7, 3) enqueue(5) – (9, 7, 3, 5) dequeue() 9 (7, 3, 5)