Professor Abdul-Quader
Lesson 11 (Shortest Path Problems)
Given a list of data, remove all duplicates from the data. That is, if your list is [1, 2, 1, 4, 3, 1, 3, 4, 1, 2], you should return [1, 2, 4, 3].
Notice that the list returned should have the elements listed in the order that they first appeared in the original list.
Given a tree (not necessarily a BST, not necessarily even a binary tree), and a node N, output the path (list of vertices visited) from the root to N.
void pathToNode(Node n):
parents[root] = null
queue.enqueue(root)
Node current = queue.dequeue()
while (current != n) {
queue.enqueue(current.left)
queue.enqueue(current.right)
parents[current.left] = current
parents[current.right] = current
current = queue.dequeue()
}
Now backtrack:
LinkedList<T> list;
Node current = n;
while (current != null) {
list.addFirst(current.data)
current = parents[current]
}
print(list)
The same algorithm can be used in more general “shortest path” problems. For example: given two positions on a chessboard, find the shortest path a knight can take to go from the starting to the ending position. Recall that knights can either move two steps horizontally and one step vertically, or two steps vertically and one horizontally.
Let’s implement the shortest path for the knight problem! (Code)
Given two positions on a chessboard, output the shortest path a knight can take to get from one to the other.
Suppose we have a list of synonymous words (each just a pair of Strings), and a list of pairs of search queries. Describe an algorithm which determines, for each pair of queries, if the queries are “synonymous”.
If our synonyms list contains:
Then the queries “hello world” and “hi earth” are synonymous. The queries “world hello” and “hi earth” are not synonymous (order matters). Your algorithm should return a list of booleans (true / false), which correspond to whether the given pair of queries is “synonymous”.
// checking two queries against each other
boolean queriesMatch = true;
String[] firstWords = query.first.split(" ");
String[] secondWords = query.second.split(" ");
if (firstWords.length != secondWords.length) {
returns.add(false); // add to list we will be returning
}
for (int i = 0; i < firstWords.length; i++) {
if (wordsAreNotEqual && wordsAreNotSynonymous) {
queriesMatch = false;
break;
}
}
returns.add(queriesMatch);
(firstWords[i], secondWords[i])
is in the list of synonyms?(secondWOrds[i], firstWords[i])
is in the list?Implement the Set interface in three ways: using an ArrayList, using an AVL tree (use the solution code, and using a regular BST (not a self balancing tree). Write a Benchmark class which has the following methods:
(continued)
insertRandoms(Set<Integer> set, int n)
: this method should insert n random numbers into the set, and output the amount of time it takes to do this (use System.nanoTime()
before and after you do the n insertions)insertInOrder(Set<Integer> set, int n)
: this method should insert all the numbers from \(0\) to \(n − 1\) into the set, and output the amount of time it takes to do this.(continued)
containsRandom(Set<Integer> set, int size, int n, int numSearches)
: this method should first insert size many random numbers into the set. Then it should do numSearches many calls to set.contains(n)
, and output the amount of time that takes.containsInOrder(Set<Integer> set, int size, int n, int numSearches)
: similar to above, but first insert the numbers \(0\) to \(size − 1\) in order into the set.(continued)
These methods just benchmark the calls to set.contains(n)
, after the inserts already happened.
After writing the Benchmark code, you must use that class to test out your different Set implementations. You then must do a write-up explaining your results. We assume that an AVL tree and a BST should be much better than an ArrayList. Does this actually happen for different types of insertions / different sizes of trees? Organize your results into tables and provide your analysis of what’s happening.
You should explore the following questions:
This project is due Friday, March 21 at 11:59 PM.
Operating systems need to solve a scheduling problem:
Suppose we are running one process and it ends. We have some list of processes that have been waiting. How do we decide which process to run next?