Instructor: Joanna Klukowska
Copyright 2020 Joanna Klukowska. Unless noted otherwise all content is released under a
Creative Commons Attribution-ShareAlike 4.0 International License.
Background image by Stewart Weiss
© Joanna Klukowska. CC-BY-SA.
Stacks are structures in which elements are always added and removed from the same end (depending on how you visualize the stack, you may wish to think of that end as the top of the stack). Stacks are last in first out (or LIFO) structures.
© Joanna Klukowska. CC-BY-SA.
public interface Stack<E> { /** Add an element to the top of the stack * @param item character to be added to the stack */ public void push ( E item ) ; /** Remove and return the element from the top of the stack * @return the element from the top of the stack or null if the stack is empty * from the stack. If stack is empty, null is returned. */ public E pop () ; /** Return the element from the top of the stack. * @return the element from the top of the stack or null if the stack is empty */ public E peek () ; /** Produces string representation of the stack. * @return Returns a String object that contains all elements stored on the stack. * The elements are separatedb y spaces. The top of the stack is the rightmost * element in the returned string. */ public String toString () ;}
© Joanna Klukowska. CC-BY-SA.
public interface Stack<E> { /** Add an element to the top of the stack * @param item character to be added to the stack */ public void push ( E item ) ; /** Remove and return the element from the top of the stack * @return the element from the top of the stack or null if the stack is empty * from the stack. If stack is empty, null is returned. */ public E pop () ; /** Return the element from the top of the stack. * @return the element from the top of the stack or null if the stack is empty */ public E peek () ; /** Produces string representation of the stack. * @return Returns a String object that contains all elements stored on the stack. * The elements are separatedb y spaces. The top of the stack is the rightmost * element in the returned string. */ public String toString () ;}
The Java's built in class implementing a stack is Stack<E>
.
© Joanna Klukowska. CC-BY-SA.
One way to implement a stack is to use the array as the underlying storage.
The decision that needs to be made is where should the top and bottom of the stack be.
© Joanna Klukowska. CC-BY-SA.
One way to implement a stack is to use the array as the underlying storage.
The decision that needs to be made is where should the top and bottom of the stack be.
As with the array-based list implementation, we also need to make sure that the array grows when the number of elements put on the stack reaches the capacity of the array.
© Joanna Klukowska. CC-BY-SA.
One way to implement a stack is to use the array as the underlying storage.
The decision that needs to be made is where should the top and bottom of the stack be.
As with the array-based list implementation, we also need to make sure that the array grows when the number of elements put on the stack reaches the capacity of the array.
capacity = 8
size = 0
© Joanna Klukowska. CC-BY-SA.
One way to implement a stack is to use the array as the underlying storage.
The decision that needs to be made is where should the top and bottom of the stack be.
As with the array-based list implementation, we also need to make sure that the array grows when the number of elements put on the stack reaches the capacity of the array.
capacity = 8
size = 1
© Joanna Klukowska. CC-BY-SA.
One way to implement a stack is to use the array as the underlying storage.
The decision that needs to be made is where should the top and bottom of the stack be.
As with the array-based list implementation, we also need to make sure that the array grows when the number of elements put on the stack reaches the capacity of the array.
capacity = 8
size = 2
© Joanna Klukowska. CC-BY-SA.
One way to implement a stack is to use the array as the underlying storage.
The decision that needs to be made is where should the top and bottom of the stack be.
As with the array-based list implementation, we also need to make sure that the array grows when the number of elements put on the stack reaches the capacity of the array.
capacity = 8
size = 3
© Joanna Klukowska. CC-BY-SA.
One way to implement a stack is to use the array as the underlying storage.
The decision that needs to be made is where should the top and bottom of the stack be.
As with the array-based list implementation, we also need to make sure that the array grows when the number of elements put on the stack reaches the capacity of the array.
capacity = 8
size = 2
The top element is at index 2, so that is the element removed when pop
is called.
© Joanna Klukowska. CC-BY-SA.
One way to implement a stack is to use the array as the underlying storage.
The decision that needs to be made is where should the top and bottom of the stack be.
As with the array-based list implementation, we also need to make sure that the array grows when the number of elements put on the stack reaches the capacity of the array.
capacity = 8
size = 5
© Joanna Klukowska. CC-BY-SA.
One way to implement a stack is to use the array as the underlying storage.
The decision that needs to be made is where should the top and bottom of the stack be.
As with the array-based list implementation, we also need to make sure that the array grows when the number of elements put on the stack reaches the capacity of the array.
capacity = 8
size = 3
© Joanna Klukowska. CC-BY-SA.
One way to implement a stack is to use the array as the underlying storage.
The decision that needs to be made is where should the top and bottom of the stack be.
As with the array-based list implementation, we also need to make sure that the array grows when the number of elements put on the stack reaches the capacity of the array.
© Joanna Klukowska. CC-BY-SA.
One way to implement a stack is to use the array as the underlying storage.
The decision that needs to be made is where should the top and bottom of the stack be.
As with the array-based list implementation, we also need to make sure that the array grows when the number of elements put on the stack reaches the capacity of the array.
bottom of the stack is always at index 0
top of the stack moves as we push and pop elements
© Joanna Klukowska. CC-BY-SA.
One way to implement a stack is to use the array as the underlying storage.
The decision that needs to be made is where should the top and bottom of the stack be.
As with the array-based list implementation, we also need to make sure that the array grows when the number of elements put on the stack reaches the capacity of the array.
bottom of the stack is always at index 0
top of the stack moves as we push and pop elements
how do we know the index at which the next element should be pushed/added?
© Joanna Klukowska. CC-BY-SA.
One way to implement a stack is to use the array as the underlying storage.
The decision that needs to be made is where should the top and bottom of the stack be.
As with the array-based list implementation, we also need to make sure that the array grows when the number of elements put on the stack reaches the capacity of the array.
bottom of the stack is always at index 0
top of the stack moves as we push and pop elements
how do we know the index at which the next element should be pushed/added? the index of the first empty space is size
© Joanna Klukowska. CC-BY-SA.
One way to implement a stack is to use the array as the underlying storage.
The decision that needs to be made is where should the top and bottom of the stack be.
As with the array-based list implementation, we also need to make sure that the array grows when the number of elements put on the stack reaches the capacity of the array.
bottom of the stack is always at index 0
top of the stack moves as we push and pop elements
how do we know the index at which the next element should be pushed/added? the index of the first empty space is size
how do we know the index from which the element should be popped/removed?
© Joanna Klukowska. CC-BY-SA.
One way to implement a stack is to use the array as the underlying storage.
The decision that needs to be made is where should the top and bottom of the stack be.
As with the array-based list implementation, we also need to make sure that the array grows when the number of elements put on the stack reaches the capacity of the array.
bottom of the stack is always at index 0
top of the stack moves as we push and pop elements
how do we know the index at which the next element should be pushed/added? the index of the first empty space is size
how do we know the index from which the element should be popped/removed? the index of the top is size - 1
© Joanna Klukowska. CC-BY-SA.
Algorithm for push
if size >= capacity grow the array and copy all the elements to the new arrayarray[size] = new elementsize++
© Joanna Klukowska. CC-BY-SA.
Algorithm for push
if size >= capacity grow the array and copy all the elements to the new arrayarray[size] = new elementsize++
Algorithm for pop
if size == 0 nothing to popelse size-- return array[size]
© Joanna Klukowska. CC-BY-SA.
Algorithm for push
if size >= capacity grow the array and copy all the elements to the new arrayarray[size] = new elementsize++
Algorithm for pop
if size == 0 nothing to popelse size-- return array[size]
Alternative algorithm for pop
if size == 0 nothing to popelse tmp = array[size-1] array[size-1] = null size-- return tmp
© Joanna Klukowska. CC-BY-SA.
Algorithm for push
if size >= capacity grow the array and copy all the elements to the new arrayarray[size] = new elementsize++
Algorithm for pop
if size == 0 nothing to popelse size-- return array[size]
Alternative algorithm for pop
if size == 0 nothing to popelse tmp = array[size-1] array[size-1] = null size-- return tmp
Algorithm for top (return, but do not remove the top element)
if size == 0 nothing to returnelse return array[size-1]
© Joanna Klukowska. CC-BY-SA.
Algorithm for push
if size >= capacity grow the array and copy all the elements to the new arrayarray[size] = new elementsize++
Algorithm for pop
if size == 0 nothing to popelse size-- return array[size]
Alternative algorithm for pop
if size == 0 nothing to popelse tmp = array[size-1] array[size-1] = null size-- return tmp
Algorithm for top (return, but do not remove the top element)
if size == 0 nothing to returnelse return array[size-1]
What is the performance of each of these algorithms?
© Joanna Klukowska. CC-BY-SA.
An alternative way of implementing a stack is to use a reference based structure just like we did for a list.
© Joanna Klukowska. CC-BY-SA.
An alternative way of implementing a stack is to use a reference based structure just like we did for a list.
Again, the decision that needs to be made is where should the top and bottom of the stack be.
© Joanna Klukowska. CC-BY-SA.
An alternative way of implementing a stack is to use a reference based structure just like we did for a list.
Again, the decision that needs to be made is where should the top and bottom of the stack be.
Which of the options A or B is better? Or is there no difference?
© Joanna Klukowska. CC-BY-SA.
© Joanna Klukowska. CC-BY-SA.
© Joanna Klukowska. CC-BY-SA.
© Joanna Klukowska. CC-BY-SA.
© Joanna Klukowska. CC-BY-SA.
© Joanna Klukowska. CC-BY-SA.
© Joanna Klukowska. CC-BY-SA.
© Joanna Klukowska. CC-BY-SA.
© Joanna Klukowska. CC-BY-SA.
© Joanna Klukowska. CC-BY-SA.
© Joanna Klukowska. CC-BY-SA.
© Joanna Klukowska. CC-BY-SA.
© Joanna Klukowska. CC-BY-SA.
© Joanna Klukowska. CC-BY-SA.
© Joanna Klukowska. CC-BY-SA.
Queues are structures in which elements are added to one end (rear/back of a queue) and removed from the other end (front of a queue). Queues are first in first out structures (FIFO).
© Joanna Klukowska. CC-BY-SA.
public interface Queue<E> { /** Add an element to the queue. * @param item an element to be added to the queue */ public void enqueue ( E item ) ; /** Remove and return the element from the front of the queue * @return the element from the front of the queue or null if queue is empty */ public E dequeue () ; /** Return the element from the front of the queue * @return the element from the front of the queue or null if queue is empty */ public E peek () ; /** Compute a string representation of the queue. * @return String object representing the queue. The string should contain the * current queue elements one per line. */ public String toString () ;}
© Joanna Klukowska. CC-BY-SA.
public interface Queue<E> { /** Add an element to the queue. * @param item an element to be added to the queue */ public void enqueue ( E item ) ; /** Remove and return the element from the front of the queue * @return the element from the front of the queue or null if queue is empty */ public E dequeue () ; /** Return the element from the front of the queue * @return the element from the front of the queue or null if queue is empty */ public E peek () ; /** Compute a string representation of the queue. * @return String object representing the queue. The string should contain the * current queue elements one per line. */ public String toString () ;}
Java has a built-in interface for a queue: Queue<E>
,
and there are several classes that implement this interface.
Note that the names for the methods are different in the above interface.
© Joanna Klukowska. CC-BY-SA.
One way to implement a queue is to use the array as the underlying storage.
The decision that needs to be made is where should the front and the end of the queue be.
© Joanna Klukowska. CC-BY-SA.
One way to implement a queue is to use the array as the underlying storage.
The decision that needs to be made is where should the front and the end of the queue be.
And as before, we also need to make sure that the array grows when the number of elements in the queue reaches the capacity of the array.
© Joanna Klukowska. CC-BY-SA.
One way to implement a queue is to use the array as the underlying storage.
The decision that needs to be made is where should the front and the end of the queue be.
And as before, we also need to make sure that the array grows when the number of elements in the queue reaches the capacity of the array.
capacity = 8
size = 0
front = 0
back = 0
© Joanna Klukowska. CC-BY-SA.
One way to implement a queue is to use the array as the underlying storage.
The decision that needs to be made is where should the front and the end of the queue be.
And as before, we also need to make sure that the array grows when the number of elements in the queue reaches the capacity of the array.
capacity = 8
size = 1
front = 0
back = 1
© Joanna Klukowska. CC-BY-SA.
One way to implement a queue is to use the array as the underlying storage.
The decision that needs to be made is where should the front and the end of the queue be.
And as before, we also need to make sure that the array grows when the number of elements in the queue reaches the capacity of the array.
capacity = 8
size = 2
front = 0
back = 2
© Joanna Klukowska. CC-BY-SA.
One way to implement a queue is to use the array as the underlying storage.
The decision that needs to be made is where should the front and the end of the queue be.
And as before, we also need to make sure that the array grows when the number of elements in the queue reaches the capacity of the array.
capacity = 8
size = 5
front = 0
back = 5
© Joanna Klukowska. CC-BY-SA.
One way to implement a queue is to use the array as the underlying storage.
The decision that needs to be made is where should the front and the end of the queue be.
And as before, we also need to make sure that the array grows when the number of elements in the queue reaches the capacity of the array.
capacity = 8
size = 4
front = 1
back = 5
© Joanna Klukowska. CC-BY-SA.
One way to implement a queue is to use the array as the underlying storage.
The decision that needs to be made is where should the front and the end of the queue be.
And as before, we also need to make sure that the array grows when the number of elements in the queue reaches the capacity of the array.
capacity = 8
size = 3
front = 2
back = 5
© Joanna Klukowska. CC-BY-SA.
One way to implement a queue is to use the array as the underlying storage.
The decision that needs to be made is where should the front and the end of the queue be.
And as before, we also need to make sure that the array grows when the number of elements in the queue reaches the capacity of the array.
capacity = 8
size = 3
front = 2
back = 5
front - index of the first element (size > 0)
back - index at which the next element should be added
size = back - front
© Joanna Klukowska. CC-BY-SA.
One way to implement a queue is to use the array as the underlying storage.
The decision that needs to be made is where should the front and the end of the queue be.
And as before, we also need to make sure that the array grows when the number of elements in the queue reaches the capacity of the array.
capacity = 8
size = 6
front = 2
back = ???
© Joanna Klukowska. CC-BY-SA.
One way to implement a queue is to use the array as the underlying storage.
The decision that needs to be made is where should the front and the end of the queue be.
And as before, we also need to make sure that the array grows when the number of elements in the queue reaches the capacity of the array.
capacity = 8
size = 6 + 1
front = 2
back = ???
© Joanna Klukowska. CC-BY-SA.
One way to implement a queue is to use the array as the underlying storage.
The decision that needs to be made is where should the front and the end of the queue be.
And as before, we also need to make sure that the array grows when the number of elements in the queue reaches the capacity of the array.
capacity = 8
size = 7
front = 2
back = 1
© Joanna Klukowska. CC-BY-SA.
One way to implement a queue is to use the array as the underlying storage.
The decision that needs to be made is where should the front and the end of the queue be.
And as before, we also need to make sure that the array grows when the number of elements in the queue reaches the capacity of the array.
capacity = 8
size = 7
front = 2
back = 1
We can use the array as a circular array: where there are empty locations available in the front of the array, we wrap the values back to the start of the array instead of allocating a new one.
© Joanna Klukowska. CC-BY-SA.
One way to implement a queue is to use the array as the underlying storage.
The decision that needs to be made is where should the front and the end of the queue be.
And as before, we also need to make sure that the array grows when the number of elements in the queue reaches the capacity of the array.
capacity = 8
size = 7
front = 2
back = 1
We can use the array as a circular array: where there are empty locations available in the front of the array, we wrap the values back to the start of the array instead of allocating a new one.
size = back - front
size =
(back - front) mod capacity
(when size < capacity)
© Joanna Klukowska. CC-BY-SA.
© Joanna Klukowska. CC-BY-SA.
10 mod 8 = 2
2 mod 8 = 2
© Joanna Klukowska. CC-BY-SA.
10 mod 8 = 2
2 mod 8 = 2
29 mod 8 = 5
© Joanna Klukowska. CC-BY-SA.
10 mod 8 = 2
2 mod 8 = 2
29 mod 8 = 5
-6 mod 8 = (-6 + 8 ) mod 8 = 2 mod 8 = 2
© Joanna Klukowska. CC-BY-SA.
10 mod 8 = 2
2 mod 8 = 2
29 mod 8 = 5
-6 mod 8 = (-6 + 8 ) mod 8 = 2 mod 8 = 2
-9 mod 8 = -1 mod 8 = 7 mod 8 = 7
© Joanna Klukowska. CC-BY-SA.
10 mod 8 = 2
2 mod 8 = 2
29 mod 8 = 5
-6 mod 8 = (-6 + 8 ) mod 8 = 2 mod 8 = 2
-9 mod 8 = -1 mod 8 = 7 mod 8 = 7
Note that Java's %
operator does not compute the true modulus, it computes
the remainder. The two are the same when both operands are positive.
In java: -6 % 8
evaluates to -6, to obtain true modulus, we need to add the divisor
to the result: (-6 % 8) + 8
.
© Joanna Klukowska. CC-BY-SA.
One way to implement a queue is to use the array as the underlying storage.
The decision that needs to be made is where should the front and the end of the queue be.
And as before, we also need to make sure that the array grows when the number of elements in the queue reaches the capacity of the array.
capacity = 8
size = 7
front = 2
back = 1
© Joanna Klukowska. CC-BY-SA.
One way to implement a queue is to use the array as the underlying storage.
The decision that needs to be made is where should the front and the end of the queue be.
And as before, we also need to make sure that the array grows when the number of elements in the queue reaches the capacity of the array.
capacity = 8
size = 8
front = 2
back = 2
© Joanna Klukowska. CC-BY-SA.
One way to implement a queue is to use the array as the underlying storage.
The decision that needs to be made is where should the front and the end of the queue be.
And as before, we also need to make sure that the array grows when the number of elements in the queue reaches the capacity of the array.
capacity = 8
size = 8
front = 2
back = 2
© Joanna Klukowska. CC-BY-SA.
One way to implement a queue is to use the array as the underlying storage.
The decision that needs to be made is where should the front and the end of the queue be.
And as before, we also need to make sure that the array grows when the number of elements in the queue reaches the capacity of the array.
capacity = 8
size = 8
front = 2
back = 2
size == capacity
so we need a larger array
© Joanna Klukowska. CC-BY-SA.
One way to implement a queue is to use the array as the underlying storage.
The decision that needs to be made is where should the front and the end of the queue be.
And as before, we also need to make sure that the array grows when the number of elements in the queue reaches the capacity of the array.
capacity = 8
size = 8
front = 2
back = 2
size == capacity
so we need a larger array
© Joanna Klukowska. CC-BY-SA.
One way to implement a queue is to use the array as the underlying storage.
The decision that needs to be made is where should the front and the end of the queue be.
And as before, we also need to make sure that the array grows when the number of elements in the queue reaches the capacity of the array.
capacity = 8
size = 8
front = 2
back = 2
size == capacity
so we need a larger array
© Joanna Klukowska. CC-BY-SA.
One way to implement a queue is to use the array as the underlying storage.
The decision that needs to be made is where should the front and the end of the queue be.
And as before, we also need to make sure that the array grows when the number of elements in the queue reaches the capacity of the array.
capacity = 8
size = 8
front = 2
back = 2
size == capacity
so we need a larger array
Will this work?
© Joanna Klukowska. CC-BY-SA.
One way to implement a queue is to use the array as the underlying storage.
The decision that needs to be made is where should the front and the end of the queue be.
And as before, we also need to make sure that the array grows when the number of elements in the queue reaches the capacity of the array.
capacity = 8
size = 8
front = 2
back = 2
size == capacity
so we need a larger array
Will this work?
NO
© Joanna Klukowska. CC-BY-SA.
One way to implement a queue is to use the array as the underlying storage.
The decision that needs to be made is where should the front and the end of the queue be.
And as before, we also need to make sure that the array grows when the number of elements in the queue reaches the capacity of the array.
capacity = 8
size = 8
front = 2
back = 2
size == capacity
so we need a larger array
© Joanna Klukowska. CC-BY-SA.
One way to implement a queue is to use the array as the underlying storage.
The decision that needs to be made is where should the front and the end of the queue be.
And as before, we also need to make sure that the array grows when the number of elements in the queue reaches the capacity of the array.
capacity = 8
size = 8
front = 2
back = 2
size == capacity
so we need a larger array
i
in the old array, copy it to index (i - front) mod capacity
© Joanna Klukowska. CC-BY-SA.
One way to implement a queue is to use the array as the underlying storage.
The decision that needs to be made is where should the front and the end of the queue be.
And as before, we also need to make sure that the array grows when the number of elements in the queue reaches the capacity of the array.
capacity = 8
size = 8
front = 2
back = 2
size == capacity
so we need a larger array
i
in the old array, copy it to index (i - front) mod capacity
Will this work?
© Joanna Klukowska. CC-BY-SA.
One way to implement a queue is to use the array as the underlying storage.
The decision that needs to be made is where should the front and the end of the queue be.
And as before, we also need to make sure that the array grows when the number of elements in the queue reaches the capacity of the array.
capacity = 8
size = 8
front = 2
back = 2
size == capacity
so we need a larger array
i
in the old array, copy it to index (i - front) mod capacity
Will this work? YES
© Joanna Klukowska. CC-BY-SA.
An alternative way of implementing a queue is to use a reference based structure just like we did for a list.
© Joanna Klukowska. CC-BY-SA.
An alternative way of implementing a queue is to use a reference based structure just like we did for a list.
Again, the decision that needs to be made is where should the front and back of the queue be.
© Joanna Klukowska. CC-BY-SA.
An alternative way of implementing a queue is to use a reference based structure just like we did for a list.
Again, the decision that needs to be made is where should the front and back of the queue be.
Which of the options A or B is better? Or is there no difference?
© Joanna Klukowska. CC-BY-SA.
© Joanna Klukowska. CC-BY-SA.
© Joanna Klukowska. CC-BY-SA.
© Joanna Klukowska. CC-BY-SA.
© Joanna Klukowska. CC-BY-SA.
© Joanna Klukowska. CC-BY-SA.
© Joanna Klukowska. CC-BY-SA.
© Joanna Klukowska. CC-BY-SA.
© Joanna Klukowska. CC-BY-SA.
© Joanna Klukowska. CC-BY-SA.
© Joanna Klukowska. CC-BY-SA.
© Joanna Klukowska. CC-BY-SA.
On slide #25 we have algorithms for the stack operations when it is implemented using an array. Create similar algorithms for the array-based queue implementation.
© Joanna Klukowska. CC-BY-SA.
Study the Stack<E>
class.
The above Stack<E>
class mentions
Deque<E>
interface.
The Queue<E>
interface is implemented by many classes. Pick the ones that you are familiar with and divide them into array-based
implementations and reference-based implementations.
Instructor: Joanna Klukowska
Copyright 2020 Joanna Klukowska. Unless noted otherwise all content is released under a
Creative Commons Attribution-ShareAlike 4.0 International License.
Background image by Stewart Weiss
© Joanna Klukowska. CC-BY-SA.
Stacks are structures in which elements are always added and removed from the same end (depending on how you visualize the stack, you may wish to think of that end as the top of the stack). Stacks are last in first out (or LIFO) structures.
© Joanna Klukowska. CC-BY-SA.
public interface Stack<E> { /** Add an element to the top of the stack * @param item character to be added to the stack */ public void push ( E item ) ; /** Remove and return the element from the top of the stack * @return the element from the top of the stack or null if the stack is empty * from the stack. If stack is empty, null is returned. */ public E pop () ; /** Return the element from the top of the stack. * @return the element from the top of the stack or null if the stack is empty */ public E peek () ; /** Produces string representation of the stack. * @return Returns a String object that contains all elements stored on the stack. * The elements are separatedb y spaces. The top of the stack is the rightmost * element in the returned string. */ public String toString () ;}
© Joanna Klukowska. CC-BY-SA.
public interface Stack<E> { /** Add an element to the top of the stack * @param item character to be added to the stack */ public void push ( E item ) ; /** Remove and return the element from the top of the stack * @return the element from the top of the stack or null if the stack is empty * from the stack. If stack is empty, null is returned. */ public E pop () ; /** Return the element from the top of the stack. * @return the element from the top of the stack or null if the stack is empty */ public E peek () ; /** Produces string representation of the stack. * @return Returns a String object that contains all elements stored on the stack. * The elements are separatedb y spaces. The top of the stack is the rightmost * element in the returned string. */ public String toString () ;}
The Java's built in class implementing a stack is Stack<E>
.
© Joanna Klukowska. CC-BY-SA.
One way to implement a stack is to use the array as the underlying storage.
The decision that needs to be made is where should the top and bottom of the stack be.
© Joanna Klukowska. CC-BY-SA.
One way to implement a stack is to use the array as the underlying storage.
The decision that needs to be made is where should the top and bottom of the stack be.
As with the array-based list implementation, we also need to make sure that the array grows when the number of elements put on the stack reaches the capacity of the array.
© Joanna Klukowska. CC-BY-SA.
One way to implement a stack is to use the array as the underlying storage.
The decision that needs to be made is where should the top and bottom of the stack be.
As with the array-based list implementation, we also need to make sure that the array grows when the number of elements put on the stack reaches the capacity of the array.
capacity = 8
size = 0
© Joanna Klukowska. CC-BY-SA.
One way to implement a stack is to use the array as the underlying storage.
The decision that needs to be made is where should the top and bottom of the stack be.
As with the array-based list implementation, we also need to make sure that the array grows when the number of elements put on the stack reaches the capacity of the array.
capacity = 8
size = 1
© Joanna Klukowska. CC-BY-SA.
One way to implement a stack is to use the array as the underlying storage.
The decision that needs to be made is where should the top and bottom of the stack be.
As with the array-based list implementation, we also need to make sure that the array grows when the number of elements put on the stack reaches the capacity of the array.
capacity = 8
size = 2
© Joanna Klukowska. CC-BY-SA.
One way to implement a stack is to use the array as the underlying storage.
The decision that needs to be made is where should the top and bottom of the stack be.
As with the array-based list implementation, we also need to make sure that the array grows when the number of elements put on the stack reaches the capacity of the array.
capacity = 8
size = 3
© Joanna Klukowska. CC-BY-SA.
One way to implement a stack is to use the array as the underlying storage.
The decision that needs to be made is where should the top and bottom of the stack be.
As with the array-based list implementation, we also need to make sure that the array grows when the number of elements put on the stack reaches the capacity of the array.
capacity = 8
size = 2
The top element is at index 2, so that is the element removed when pop
is called.
© Joanna Klukowska. CC-BY-SA.
One way to implement a stack is to use the array as the underlying storage.
The decision that needs to be made is where should the top and bottom of the stack be.
As with the array-based list implementation, we also need to make sure that the array grows when the number of elements put on the stack reaches the capacity of the array.
capacity = 8
size = 5
© Joanna Klukowska. CC-BY-SA.
One way to implement a stack is to use the array as the underlying storage.
The decision that needs to be made is where should the top and bottom of the stack be.
As with the array-based list implementation, we also need to make sure that the array grows when the number of elements put on the stack reaches the capacity of the array.
capacity = 8
size = 3
© Joanna Klukowska. CC-BY-SA.
One way to implement a stack is to use the array as the underlying storage.
The decision that needs to be made is where should the top and bottom of the stack be.
As with the array-based list implementation, we also need to make sure that the array grows when the number of elements put on the stack reaches the capacity of the array.
© Joanna Klukowska. CC-BY-SA.
One way to implement a stack is to use the array as the underlying storage.
The decision that needs to be made is where should the top and bottom of the stack be.
As with the array-based list implementation, we also need to make sure that the array grows when the number of elements put on the stack reaches the capacity of the array.
bottom of the stack is always at index 0
top of the stack moves as we push and pop elements
© Joanna Klukowska. CC-BY-SA.
One way to implement a stack is to use the array as the underlying storage.
The decision that needs to be made is where should the top and bottom of the stack be.
As with the array-based list implementation, we also need to make sure that the array grows when the number of elements put on the stack reaches the capacity of the array.
bottom of the stack is always at index 0
top of the stack moves as we push and pop elements
how do we know the index at which the next element should be pushed/added?
© Joanna Klukowska. CC-BY-SA.
One way to implement a stack is to use the array as the underlying storage.
The decision that needs to be made is where should the top and bottom of the stack be.
As with the array-based list implementation, we also need to make sure that the array grows when the number of elements put on the stack reaches the capacity of the array.
bottom of the stack is always at index 0
top of the stack moves as we push and pop elements
how do we know the index at which the next element should be pushed/added? the index of the first empty space is size
© Joanna Klukowska. CC-BY-SA.
One way to implement a stack is to use the array as the underlying storage.
The decision that needs to be made is where should the top and bottom of the stack be.
As with the array-based list implementation, we also need to make sure that the array grows when the number of elements put on the stack reaches the capacity of the array.
bottom of the stack is always at index 0
top of the stack moves as we push and pop elements
how do we know the index at which the next element should be pushed/added? the index of the first empty space is size
how do we know the index from which the element should be popped/removed?
© Joanna Klukowska. CC-BY-SA.
One way to implement a stack is to use the array as the underlying storage.
The decision that needs to be made is where should the top and bottom of the stack be.
As with the array-based list implementation, we also need to make sure that the array grows when the number of elements put on the stack reaches the capacity of the array.
bottom of the stack is always at index 0
top of the stack moves as we push and pop elements
how do we know the index at which the next element should be pushed/added? the index of the first empty space is size
how do we know the index from which the element should be popped/removed? the index of the top is size - 1
© Joanna Klukowska. CC-BY-SA.
Algorithm for push
if size >= capacity grow the array and copy all the elements to the new arrayarray[size] = new elementsize++
© Joanna Klukowska. CC-BY-SA.
Algorithm for push
if size >= capacity grow the array and copy all the elements to the new arrayarray[size] = new elementsize++
Algorithm for pop
if size == 0 nothing to popelse size-- return array[size]
© Joanna Klukowska. CC-BY-SA.
Algorithm for push
if size >= capacity grow the array and copy all the elements to the new arrayarray[size] = new elementsize++
Algorithm for pop
if size == 0 nothing to popelse size-- return array[size]
Alternative algorithm for pop
if size == 0 nothing to popelse tmp = array[size-1] array[size-1] = null size-- return tmp
© Joanna Klukowska. CC-BY-SA.
Algorithm for push
if size >= capacity grow the array and copy all the elements to the new arrayarray[size] = new elementsize++
Algorithm for pop
if size == 0 nothing to popelse size-- return array[size]
Alternative algorithm for pop
if size == 0 nothing to popelse tmp = array[size-1] array[size-1] = null size-- return tmp
Algorithm for top (return, but do not remove the top element)
if size == 0 nothing to returnelse return array[size-1]
© Joanna Klukowska. CC-BY-SA.
Algorithm for push
if size >= capacity grow the array and copy all the elements to the new arrayarray[size] = new elementsize++
Algorithm for pop
if size == 0 nothing to popelse size-- return array[size]
Alternative algorithm for pop
if size == 0 nothing to popelse tmp = array[size-1] array[size-1] = null size-- return tmp
Algorithm for top (return, but do not remove the top element)
if size == 0 nothing to returnelse return array[size-1]
What is the performance of each of these algorithms?
© Joanna Klukowska. CC-BY-SA.
An alternative way of implementing a stack is to use a reference based structure just like we did for a list.
© Joanna Klukowska. CC-BY-SA.
An alternative way of implementing a stack is to use a reference based structure just like we did for a list.
Again, the decision that needs to be made is where should the top and bottom of the stack be.
© Joanna Klukowska. CC-BY-SA.
An alternative way of implementing a stack is to use a reference based structure just like we did for a list.
Again, the decision that needs to be made is where should the top and bottom of the stack be.
Which of the options A or B is better? Or is there no difference?
© Joanna Klukowska. CC-BY-SA.
© Joanna Klukowska. CC-BY-SA.
© Joanna Klukowska. CC-BY-SA.
© Joanna Klukowska. CC-BY-SA.
© Joanna Klukowska. CC-BY-SA.
© Joanna Klukowska. CC-BY-SA.
© Joanna Klukowska. CC-BY-SA.
© Joanna Klukowska. CC-BY-SA.
© Joanna Klukowska. CC-BY-SA.
© Joanna Klukowska. CC-BY-SA.
© Joanna Klukowska. CC-BY-SA.
© Joanna Klukowska. CC-BY-SA.
© Joanna Klukowska. CC-BY-SA.
© Joanna Klukowska. CC-BY-SA.
© Joanna Klukowska. CC-BY-SA.
Queues are structures in which elements are added to one end (rear/back of a queue) and removed from the other end (front of a queue). Queues are first in first out structures (FIFO).
© Joanna Klukowska. CC-BY-SA.
public interface Queue<E> { /** Add an element to the queue. * @param item an element to be added to the queue */ public void enqueue ( E item ) ; /** Remove and return the element from the front of the queue * @return the element from the front of the queue or null if queue is empty */ public E dequeue () ; /** Return the element from the front of the queue * @return the element from the front of the queue or null if queue is empty */ public E peek () ; /** Compute a string representation of the queue. * @return String object representing the queue. The string should contain the * current queue elements one per line. */ public String toString () ;}
© Joanna Klukowska. CC-BY-SA.
public interface Queue<E> { /** Add an element to the queue. * @param item an element to be added to the queue */ public void enqueue ( E item ) ; /** Remove and return the element from the front of the queue * @return the element from the front of the queue or null if queue is empty */ public E dequeue () ; /** Return the element from the front of the queue * @return the element from the front of the queue or null if queue is empty */ public E peek () ; /** Compute a string representation of the queue. * @return String object representing the queue. The string should contain the * current queue elements one per line. */ public String toString () ;}
Java has a built-in interface for a queue: Queue<E>
,
and there are several classes that implement this interface.
Note that the names for the methods are different in the above interface.
© Joanna Klukowska. CC-BY-SA.
One way to implement a queue is to use the array as the underlying storage.
The decision that needs to be made is where should the front and the end of the queue be.
© Joanna Klukowska. CC-BY-SA.
One way to implement a queue is to use the array as the underlying storage.
The decision that needs to be made is where should the front and the end of the queue be.
And as before, we also need to make sure that the array grows when the number of elements in the queue reaches the capacity of the array.
© Joanna Klukowska. CC-BY-SA.
One way to implement a queue is to use the array as the underlying storage.
The decision that needs to be made is where should the front and the end of the queue be.
And as before, we also need to make sure that the array grows when the number of elements in the queue reaches the capacity of the array.
capacity = 8
size = 0
front = 0
back = 0
© Joanna Klukowska. CC-BY-SA.
One way to implement a queue is to use the array as the underlying storage.
The decision that needs to be made is where should the front and the end of the queue be.
And as before, we also need to make sure that the array grows when the number of elements in the queue reaches the capacity of the array.
capacity = 8
size = 1
front = 0
back = 1
© Joanna Klukowska. CC-BY-SA.
One way to implement a queue is to use the array as the underlying storage.
The decision that needs to be made is where should the front and the end of the queue be.
And as before, we also need to make sure that the array grows when the number of elements in the queue reaches the capacity of the array.
capacity = 8
size = 2
front = 0
back = 2
© Joanna Klukowska. CC-BY-SA.
One way to implement a queue is to use the array as the underlying storage.
The decision that needs to be made is where should the front and the end of the queue be.
And as before, we also need to make sure that the array grows when the number of elements in the queue reaches the capacity of the array.
capacity = 8
size = 5
front = 0
back = 5
© Joanna Klukowska. CC-BY-SA.
One way to implement a queue is to use the array as the underlying storage.
The decision that needs to be made is where should the front and the end of the queue be.
And as before, we also need to make sure that the array grows when the number of elements in the queue reaches the capacity of the array.
capacity = 8
size = 4
front = 1
back = 5
© Joanna Klukowska. CC-BY-SA.
One way to implement a queue is to use the array as the underlying storage.
The decision that needs to be made is where should the front and the end of the queue be.
And as before, we also need to make sure that the array grows when the number of elements in the queue reaches the capacity of the array.
capacity = 8
size = 3
front = 2
back = 5
© Joanna Klukowska. CC-BY-SA.
One way to implement a queue is to use the array as the underlying storage.
The decision that needs to be made is where should the front and the end of the queue be.
And as before, we also need to make sure that the array grows when the number of elements in the queue reaches the capacity of the array.
capacity = 8
size = 3
front = 2
back = 5
front - index of the first element (size > 0)
back - index at which the next element should be added
size = back - front
© Joanna Klukowska. CC-BY-SA.
One way to implement a queue is to use the array as the underlying storage.
The decision that needs to be made is where should the front and the end of the queue be.
And as before, we also need to make sure that the array grows when the number of elements in the queue reaches the capacity of the array.
capacity = 8
size = 6
front = 2
back = ???
© Joanna Klukowska. CC-BY-SA.
One way to implement a queue is to use the array as the underlying storage.
The decision that needs to be made is where should the front and the end of the queue be.
And as before, we also need to make sure that the array grows when the number of elements in the queue reaches the capacity of the array.
capacity = 8
size = 6 + 1
front = 2
back = ???
© Joanna Klukowska. CC-BY-SA.
One way to implement a queue is to use the array as the underlying storage.
The decision that needs to be made is where should the front and the end of the queue be.
And as before, we also need to make sure that the array grows when the number of elements in the queue reaches the capacity of the array.
capacity = 8
size = 7
front = 2
back = 1
© Joanna Klukowska. CC-BY-SA.
One way to implement a queue is to use the array as the underlying storage.
The decision that needs to be made is where should the front and the end of the queue be.
And as before, we also need to make sure that the array grows when the number of elements in the queue reaches the capacity of the array.
capacity = 8
size = 7
front = 2
back = 1
We can use the array as a circular array: where there are empty locations available in the front of the array, we wrap the values back to the start of the array instead of allocating a new one.
© Joanna Klukowska. CC-BY-SA.
One way to implement a queue is to use the array as the underlying storage.
The decision that needs to be made is where should the front and the end of the queue be.
And as before, we also need to make sure that the array grows when the number of elements in the queue reaches the capacity of the array.
capacity = 8
size = 7
front = 2
back = 1
We can use the array as a circular array: where there are empty locations available in the front of the array, we wrap the values back to the start of the array instead of allocating a new one.
size = back - front
size =
(back - front) mod capacity
(when size < capacity)
© Joanna Klukowska. CC-BY-SA.
© Joanna Klukowska. CC-BY-SA.
10 mod 8 = 2
2 mod 8 = 2
© Joanna Klukowska. CC-BY-SA.
10 mod 8 = 2
2 mod 8 = 2
29 mod 8 = 5
© Joanna Klukowska. CC-BY-SA.
10 mod 8 = 2
2 mod 8 = 2
29 mod 8 = 5
-6 mod 8 = (-6 + 8 ) mod 8 = 2 mod 8 = 2
© Joanna Klukowska. CC-BY-SA.
10 mod 8 = 2
2 mod 8 = 2
29 mod 8 = 5
-6 mod 8 = (-6 + 8 ) mod 8 = 2 mod 8 = 2
-9 mod 8 = -1 mod 8 = 7 mod 8 = 7
© Joanna Klukowska. CC-BY-SA.
10 mod 8 = 2
2 mod 8 = 2
29 mod 8 = 5
-6 mod 8 = (-6 + 8 ) mod 8 = 2 mod 8 = 2
-9 mod 8 = -1 mod 8 = 7 mod 8 = 7
Note that Java's %
operator does not compute the true modulus, it computes
the remainder. The two are the same when both operands are positive.
In java: -6 % 8
evaluates to -6, to obtain true modulus, we need to add the divisor
to the result: (-6 % 8) + 8
.
© Joanna Klukowska. CC-BY-SA.
One way to implement a queue is to use the array as the underlying storage.
The decision that needs to be made is where should the front and the end of the queue be.
And as before, we also need to make sure that the array grows when the number of elements in the queue reaches the capacity of the array.
capacity = 8
size = 7
front = 2
back = 1
© Joanna Klukowska. CC-BY-SA.
One way to implement a queue is to use the array as the underlying storage.
The decision that needs to be made is where should the front and the end of the queue be.
And as before, we also need to make sure that the array grows when the number of elements in the queue reaches the capacity of the array.
capacity = 8
size = 8
front = 2
back = 2
© Joanna Klukowska. CC-BY-SA.
One way to implement a queue is to use the array as the underlying storage.
The decision that needs to be made is where should the front and the end of the queue be.
And as before, we also need to make sure that the array grows when the number of elements in the queue reaches the capacity of the array.
capacity = 8
size = 8
front = 2
back = 2
© Joanna Klukowska. CC-BY-SA.
One way to implement a queue is to use the array as the underlying storage.
The decision that needs to be made is where should the front and the end of the queue be.
And as before, we also need to make sure that the array grows when the number of elements in the queue reaches the capacity of the array.
capacity = 8
size = 8
front = 2
back = 2
size == capacity
so we need a larger array
© Joanna Klukowska. CC-BY-SA.
One way to implement a queue is to use the array as the underlying storage.
The decision that needs to be made is where should the front and the end of the queue be.
And as before, we also need to make sure that the array grows when the number of elements in the queue reaches the capacity of the array.
capacity = 8
size = 8
front = 2
back = 2
size == capacity
so we need a larger array
© Joanna Klukowska. CC-BY-SA.
One way to implement a queue is to use the array as the underlying storage.
The decision that needs to be made is where should the front and the end of the queue be.
And as before, we also need to make sure that the array grows when the number of elements in the queue reaches the capacity of the array.
capacity = 8
size = 8
front = 2
back = 2
size == capacity
so we need a larger array
© Joanna Klukowska. CC-BY-SA.
One way to implement a queue is to use the array as the underlying storage.
The decision that needs to be made is where should the front and the end of the queue be.
And as before, we also need to make sure that the array grows when the number of elements in the queue reaches the capacity of the array.
capacity = 8
size = 8
front = 2
back = 2
size == capacity
so we need a larger array
Will this work?
© Joanna Klukowska. CC-BY-SA.
One way to implement a queue is to use the array as the underlying storage.
The decision that needs to be made is where should the front and the end of the queue be.
And as before, we also need to make sure that the array grows when the number of elements in the queue reaches the capacity of the array.
capacity = 8
size = 8
front = 2
back = 2
size == capacity
so we need a larger array
Will this work?
NO
© Joanna Klukowska. CC-BY-SA.
One way to implement a queue is to use the array as the underlying storage.
The decision that needs to be made is where should the front and the end of the queue be.
And as before, we also need to make sure that the array grows when the number of elements in the queue reaches the capacity of the array.
capacity = 8
size = 8
front = 2
back = 2
size == capacity
so we need a larger array
© Joanna Klukowska. CC-BY-SA.
One way to implement a queue is to use the array as the underlying storage.
The decision that needs to be made is where should the front and the end of the queue be.
And as before, we also need to make sure that the array grows when the number of elements in the queue reaches the capacity of the array.
capacity = 8
size = 8
front = 2
back = 2
size == capacity
so we need a larger array
i
in the old array, copy it to index (i - front) mod capacity
© Joanna Klukowska. CC-BY-SA.
One way to implement a queue is to use the array as the underlying storage.
The decision that needs to be made is where should the front and the end of the queue be.
And as before, we also need to make sure that the array grows when the number of elements in the queue reaches the capacity of the array.
capacity = 8
size = 8
front = 2
back = 2
size == capacity
so we need a larger array
i
in the old array, copy it to index (i - front) mod capacity
Will this work?
© Joanna Klukowska. CC-BY-SA.
One way to implement a queue is to use the array as the underlying storage.
The decision that needs to be made is where should the front and the end of the queue be.
And as before, we also need to make sure that the array grows when the number of elements in the queue reaches the capacity of the array.
capacity = 8
size = 8
front = 2
back = 2
size == capacity
so we need a larger array
i
in the old array, copy it to index (i - front) mod capacity
Will this work? YES
© Joanna Klukowska. CC-BY-SA.
An alternative way of implementing a queue is to use a reference based structure just like we did for a list.
© Joanna Klukowska. CC-BY-SA.
An alternative way of implementing a queue is to use a reference based structure just like we did for a list.
Again, the decision that needs to be made is where should the front and back of the queue be.
© Joanna Klukowska. CC-BY-SA.
An alternative way of implementing a queue is to use a reference based structure just like we did for a list.
Again, the decision that needs to be made is where should the front and back of the queue be.
Which of the options A or B is better? Or is there no difference?
© Joanna Klukowska. CC-BY-SA.
© Joanna Klukowska. CC-BY-SA.
© Joanna Klukowska. CC-BY-SA.
© Joanna Klukowska. CC-BY-SA.
© Joanna Klukowska. CC-BY-SA.
© Joanna Klukowska. CC-BY-SA.
© Joanna Klukowska. CC-BY-SA.
© Joanna Klukowska. CC-BY-SA.
© Joanna Klukowska. CC-BY-SA.
© Joanna Klukowska. CC-BY-SA.
© Joanna Klukowska. CC-BY-SA.
© Joanna Klukowska. CC-BY-SA.
On slide #25 we have algorithms for the stack operations when it is implemented using an array. Create similar algorithms for the array-based queue implementation.
© Joanna Klukowska. CC-BY-SA.
Study the Stack<E>
class.
The above Stack<E>
class mentions
Deque<E>
interface.
The Queue<E>
interface is implemented by many classes. Pick the ones that you are familiar with and divide them into array-based
implementations and reference-based implementations.
Keyboard shortcuts
↑, ←, Pg Up, k | Go to previous slide |
↓, →, Pg Dn, Space, j | Go to next slide |
Home | Go to first slide |
End | Go to last slide |
Number + Return | Go to specific slide |
b / m / f | Toggle blackout / mirrored / fullscreen mode |
c | Clone slideshow |
p | Toggle presenter mode |
t | Restart the presentation timer |
?, h | Toggle this help |
Esc | Back to slideshow |
Keyboard shortcuts
↑, ←, Pg Up, k | Go to previous slide |
↓, →, Pg Dn, Space, j | Go to next slide |
Home | Go to first slide |
End | Go to last slide |
Number + Return | Go to specific slide |
b / m / f | Toggle blackout / mirrored / fullscreen mode |
c | Clone slideshow |
p | Toggle presenter mode |
t | Restart the presentation timer |
?, h | Toggle this help |
Esc | Back to slideshow |