Lecture notes

advertisement
The Dictionary ADT
Definition A dictionary is an ordered or unordered list of key-element pairs,
where keys are used to locate elements in the list.
Example: consider a data structure that stores bank accounts; it can be viewed
as a dictionary, where account numbers serve as keys for identification of
account objects.
Operations (methods) on dictionaries:
size ()
Returns the size of the dictionary
empty ()
Returns true is the dictionary is empty
findItem (key)
Locates the item with the specified key. If
no such key exists, sentinel value NO_SUCH_KEY is returned. If more
than one item with the specified key exists, an arbitrary item is returned.
findAllItems (key)
Locates all items with the specified key. If
no such key exists, sentinel value NO_SUCH_KEY is returned.
removeItem (key)
Removes the item with the specified key
removeAllItems (key)
Removes all items with the specified key
insertItem (key, element)
Inserts a new key-element pair
Additional methods for ordered dictionaries
closestKeyBefore (key)
closestElemBefore (key)
closestKeyAfter (key)
closestElemAfter (key)
Returns the key of the item with largest key
less than or equal to key
Returns the element for the item with largest
key less than or equal to key
Returns the key of the item with smallest
key greater than or equal to key
Returns the element for the item with smallest
key greater than or equal to key
Sentinel value NO_SUCH_KEY is always returned if no item in the dictionary
satisfies the query.
Note Java has a built-in abstract class java.util.Dictionary In this class,
however, having two items with the same key is not allowed. If an application
assumes more than one item with the same key, an extended version of the
Dictionary class is required.
Example of unordered dictionary
Consider an empty unordered dictionary and the following set of operations:
Operation
insertItem(5,A)
insertItem(7,B)
insertItem(2,C)
insertItem(8,D)
insertItem(2,E)
findItem(7)
findItem(4)
findItem(2)
findAllItems(2)
size()
removeItem(5)
removeAllItems(2)
findItem(4)
Dictionary
Output
{(5,A)}
{(5,A), (7,B)}
{(5,A), (7,B), (2,C)}
{(5,A), (7,B), (2,C), (8,D)}
{(5,A), (7,B), (2,C), (8,D), (2,E)}
{(5,A), (7,B), (2,C), (8,D), (2,E)}
B
{(5,A), (7,B), (2,C), (8,D), (2,E)} NO_SUCH_KEY
{(5,A), (7,B), (2,C), (8,D), (2,E)}
C
{(5,A), (7,B), (2,C), (8,D), (2,E)}
C, E
{(5,A), (7,B), (2,C), (8,D), (2,E)}
5
{(7,B), (2,C), (8,D), (2,E)}
A
{(7,B), (8,D)}
C, E
{(7,B), (8,D)}
NO_SUCH_KEY
Example of ordered dictionary
Consider an empty ordered dictionary and the following set of operations:
Operation
insertItem(5,A)
insertItem(7,B)
insertItem(2,C)
insertItem(8,D)
insertItem(2,E)
findItem(7)
findItem(4)
findItem(2)
findAllItems(2)
size()
removeItem(5)
removeAllItems(2)
findItem(4)
Dictionary
Output
{(5,A)}
{(5,A), (7,B)}
{(2,C), (5,A), (7,B)}
{(2,C), (5,A), (7,B), (8,D)}
{(2,C), (2,E), (5,A), (7,B), (8,D)}
{(2,C), (2,E), (5,A), (7,B), (8,D)}
B
{(2,C), (2,E), (5,A), (7,B), (8,D)} NO_SUCH_KEY
{(2,C), (2,E), (5,A), (7,B), (8,D)}
C
{(2,C), (2,E), (5,A), (7,B), (8,D)}
C, E
{(2,C), (2,E), (5,A), (7,B), (8,D)}
5
{(2,C), (2,E), (7,B), (8,D)}
A
{(7,B), (8,D)}
C, E
{(7,B), (8,D)}
NO_SUCH_KEY
Implementations of the Dictionary ADT
Dictionaries are ordered or unordered lists. The easiest way to implement a list
is by means of an ordered or unordered sequence.
Unordered sequence implementation Items are added to the initially empty
dictionary as they arrive. insertItem(key, element) method is O(1) no matter whether the
new item is added at the beginning or at the end of the dictionary. findItem(key),
findAllItems(key), removeItem(key) and removeAllItems(key) methods, however, have
O(n) efficiency. Therefore, this implementation is appropriate in applications where the
number of insertions is very large in comparison to the number of searches and removals.
Ordered sequence implementation Items are added to the initially empty
dictionary in nondecreasing order of their keys. insertItem(key, element) method is O(n),
because a search for the proper place of the item is required. If the sequence is implemented
as an ordered array, removeItem(key) and removeAllItems(key) take O(n) time, because
all items following the item removed must be shifted to fill in the gap. If the sequence is
implemented as a doubly linked list , all methods involving search also take O(n) time.
Therefore, this implementation is inferior compared to unordered sequence implementation.
However, the efficiency of the search operation can be considerably improved, in which case
an ordered sequence implementation will become a better choice.
Implementations of the Dictionary ADT (contd.)
Array-based ranked sequence implementation A search for an item in a
sequence by its rank takes O(1) time. We can improve search efficiency in an
ordered dictionary by using binary search; thus improving the run time efficiency
of insertItem(key, element), removeItem(key) and removeAllItems(key) to
O(log n).
More efficient implementations of an ordered dictionary are binary search trees
and AVL trees which are binary search trees of a special type. The best way to
implement an unordered dictionary is by means of a hash table. We discuss AVL
trees and hash tables next.
AVL trees
Definition An AVL tree is a binary tree with an ordering property where the
heights of the children of every internal node differ by at most 1.
Example
44 (4)
17 (2)
78 (3)
32 (1)
50 (2)
48 (1)
88 (1)
62 (1)
Note: 1. Every subtree of an AVL tree is also an AVL tree.
2. The height of an AVL tree storing n keys is O(log n).
Insertion of new nodes in AVL trees
Assume you want to insert 54 in our example tree.
Step 1: Search for 54 (as if it were a binary search tree), and find where the
search terminates unsuccessfully
44 (5)
17 (2)
78 (4)
32 (1)
50 (3)
48 (1)
88 (1)
62 (2)
54 (1)
Step 2: Restore the balance of the tree.
These two children
are unbalanced
Rotation of AVL tree nodes
To restore the balance of the tree, we perform the following restructuring. Let z be the first
“unbalanced” node on the path from the newly inserted node to the root, y be the child of z
with higher height, and x be the child of y (x may be the newly inserted node). Since z became
unbalanced because of the insertion in the subtree rooted at its child y, the height of y is 2
greater than its sibling.
Let us rename nodes x, y, and z as a, b, and c, such that a precedes b and b precedes c in
inorder traversal of the currently unbalanced tree. There are 4 ways to map x, y, and z to
a, b, and c, as follows:
z=a
y=b
y=b
T0
x=c
z=a
x=c
T1
T2
T3
T0
T1
T2
T3
Rotation of AVL tree nodes (contd.)
z=c
y=b
y=b
x=a
T3
x=a
z=c
T2
T0
T1
z=a
T0
T1
y=c
T0
T2
T3
x=b
x=b
z=a
y=c
T3
T1
T2
T0
T1
T2
T3
Rotation of AVL tree nodes (contd.)
z=c
y=a
x=b
x=b
T3
y=a
z=c
T0
T1
T2
T0
T1
T2
T3
The restructure algorithm
Algorithm restructure(x):
Input: A node x that has a parent node y, and a grandparent node z.
Output: Tree involving nodes x, y and z restructured.
1. Let (a,b,c) be inorder listing of nodes x, y and z, and let (T0, T1, T2, T3) be
inorder listing of the four children subtrees of x,y, and z.
2. Replace the subtree rooted at z with a new subtree rooted at b.
3. Let a be the left child of b and let T0 and T1 be the left and right subtrees of
a, respectively.
4. Let c be the right child of b and let T2 and T3 be the left and right subtrees of
c, respectively.
If y = b, we have a single rotation, where y is rotated over z. If x = b, we have a
double rotation, where x is first rotated over y, and then over z.
Deletion of AVL tree nodes
Consider our example tree and assume that we want to delete 32.
44 (4)
17 (1)
78 (3)
50 (2)
48 (1)
These children are
unbalanced
88 (1)
62 (1)
Note: Search for the node to delete is performed as in the binary search tree.
To restore the balance of the tree, we may have to perform more than one rotation
when we move towards the root (one rotation may not be sufficient here).
Deletion of AVL tree nodes (contd.)
After the restructuring of the tree rooted in node 44:
44 (4) z=a
17 (1)
50
78 (3) y=c
x=b 50 (2)
48 (1)
88 (1)
62 (1)
44
17
78
48
62
88
Implementation of unordered dictionaries: hash
tables
Hashing is a method for directly referencing an element in a table by performing
arithmetic transformations on keys into table addresses. This is carried out in two
steps:
Step 1: Computing the so-called hash function H: K -> A.
K1
K2
K3
...
Kn
A1
A2
...
An
Step 2: Collision resolution, which handles cases where two or more different keys
hash to the same table address.
Implementation of hash tables
Hash tables consist of two components: a bucket array and a hash function.
Consider a dictionary, where keys are integers in the range [0, N-1]. Then, an
array of size N can be used to represent the dictionary. Each entry in this array is
thought of as a “bucket” (which is why we call it a “bucket array”). An element e
with key k is inserted in A[k]. Bucket entries associated with keys not present in
the dictionary contain a special NO_SUCH_KEY object. If the dictionary contains
elements with the same key, then two or more different elements may be mapped
to the same bucket of A. In this case, we say that a collision between these
elements has occurred. One easy way to deal with collisions is to allow a sequence
of elements with the same key, k, to be stored in A[k]. Assuming that an arbitrary
element with key k satisfies queries findItem(k) and removeItem(k), these
operations are now performed in O(1) time, while insertItem(k, e) needs only to
find where on the existing list A[k] to insert the new item, e. The drawback of this is
that the size of the bucket array is the size of the set from which key are drawn,
which may be huge.
Hash functions
We can limit the size of the bucket array to almost any size; however, we must
provide a way to map key values into array index values. This is done by an
appropriately selected hash function, h(k). The simplest hash function is
h(k) = k mod N
where k can be very large, while N can be as small as we want it to be. That is,
the hush function converts a large number (the key) into a smaller number
serving as an index in the bucket array.
Example. Consider the following list of keys: 10, 20, 30, 40,..., 220.
Let us consider two different sizes of the bucket array:
(1) a bucket array of size 10, and
(2) a bucket array of size 11.
Example (contd.)
Case 1:
Position
0
1
2
3
4
5
6
7
8
9
Case 2:
Key
10, 20, 30,..., 220
Position
0
1
2
3
4
5
6
7
8
9
10
Key
110, 220
100, 210
90, 200
80, 190
70, 180
60, 170
50, 160
40, 150
30, 140
20, 130
10, 120
Example 2
Consider a dictionary of strings of characters from a to z. Assume that each
character is encoded by means of 5 bits, i.e.
character
a
b
c
d
e
......
k
......
y
code
00001
00010
00011
00100
00101
01011
11001
Then, the string akey has the following code
(00001 01011 00101 11001)2 =
(44217)10
Assume that our hash table has 101 buckets. Then,
h(44217) = 44217 mod 101 = 80
That is, the key of the string akey hashes to position 80. If you do the same with
the string barh, you will see that it hashes to the same position, 80.
Hash functions (contd.)
These examples suggest that if N is a prime number, the hash function helps
spread out the distribution of hashed values. If dictionary elements are spread
fairly evenly in the hash table, the expected running times of operations
findItem, insertItem and removeItem are O(n/N), where n is the number of
elements in the dictionary, and N is the size of the bucket array. These efficiencies
are ever better, O(1), if no collision occurs (in which case only a call to the hash
function and a single array reference are needed to insert or find an item).
Collision resolution
There are 2 main ways to perform collision resolution:
1 Open addressing.
2 Chaining.
In our examples, we have assumed that collision resolution is performed by
chaining, i.e. traversing the linked list holding items with the same key in order to
find the one we are searching for, or insert a new item with that key.
In open addressing we deal with collision by finding another, unoccupied location
elsewhere in the array. The easiest way to find such a location is called linear
probing. The idea is the following. If a collision occurs when we are inserting a
new item into a table, we simply probe forward in the array, one step at a time,
until we find an empty slot where to store the new item. When we remove an item,
we start by calculating the hash function and test the identified index location. If
the item is not there, we examine each array entry from the index location until:
(1) the item is found; (2) an empty location is encountered, or (3) the array end is
reached.
Download