This assignment lays the groundwork for an application you’ll build in a later homework assignment. This assignment has two main parts. In the first part, you will make your graph class(es) generic. In the second part, you will implement a different path-finding algorithm for graphs known as Dijkstra’s algorithm.
Augmenting Your Graph and Marvel Paths
Problem 1: Making Your Graph Generic
In the application you’ll be building in subsequent homework, your mission is to find the shortest route to visit a certain set of buildings. A graph is an excellent representation of a map, and luckily you have already specified and implemented a graph. Unfortunately, your graph only stores Strings, whereas the route-finding application needs to store non-String data types in the nodes and edges. More generally, your graph would be much more widely useful if only the client could choose the data types to be stored in nodes and edges. Herein lies the power of generics!
Your task is to convert your graph ADT to a generic class. Rather than always storing the data in nodes and edge labels as Strings, it should have two type parameters representing the data types to be stored in nodes and edges. Directly modify your existing classes under hw4 — there is no need to copy or duplicate code.
When you are done, your previously-written HW4 and HW5 tests and MarvelPaths will no longer compile. Modify these classes to construct and use graph objects parameterized with Strings. All code must compile and all tests must pass when you submit your homework. Depending on your changes, some of your tests may no longer be valid. Try to adapt your tests to your new implementation, or discard them and write the new ones: they should help you build confidence in your implementation. But, don’t overdo it: as with any testing, stop when you feel that the additional effort is not being repaid in terms of increased confidence in your implementation.
Build tools and generic code
We want you to configure Eclipse to show generics problems as errors. By default, Eclipse shows generics problems as warnings (indicated by yellow lines and markers). You can configure Eclipse to instead issue errors (indicated by red lines and markers) for these problems. Doing so will help you remember to write acceptable generics code.
To make this configuration, go to Eclipse >> Preferences and select Java >> Compiler >> Errors/Warnings. Under Generic types, change the value of Unchecked generic type operation to Error.
(Note that there is another setting named “Usage of a raw type” that is set to Ignore by default. We recommend that this option be set to Warn because it is specific to the Eclipse compiler and checks for more stringent requirements than required by the Java language specification.)
Hint: Sometimes you may find that classes which previously compiled are now reporting “[some class] cannot be resolved” errors in Eclipse. You can fix these errors by doing a clean build: go to Project >> Properties >> Clean…, select your csci2600 project, and hit OK.
Problem 2: Weighted Graphs and Least-Cost Paths
In a weighted graph, the label on each edge is a length, cost, or weight representing the cost of traversing that edge. Depending on the application, the cost may be measured in time, money, physical distance, etc. The total cost of a path is the sum of the costs of all edges in that path, and the minimum-cost path between two nodes is the path with the lowest total cost between those nodes.
You will implement Dijkstra’s algorithm, which finds a minimum-cost path between two given nodes in a graph with all nonnegative edge weights. Below is a pseudocode algorithm that you may use in your implementation. You are free to modify it as long as you are essentially still implementing Dijkstra’s algorithm. Your implementation of the algorithm may assume a graph with Double edge weights.
The algorithm uses a priority queue. The standard Java libraries include an implementation of a PriorityQueue.
Dijkstra's algorithm assumes a graph with all nonnegative edge weights.
start = starting node dest = destination node
active = priority queue. Each element is a path from start to a given node.
A path's "priority" in the queue is the total cost of that path.
Nodes for which no path is known yet are not in
the queue. finished = set of nodes for which we know the minimum-
cost path from start.
// Initially we only know of the path from start to itself, which has a cost
// of zero because it contains no edges. Add a path from start to itself to active
while active is non-empty: // minPath is the lowest-cost path in active and is
the minimum-cost // path for some node
minPath = active.removeMin() minDest = destination node in minPath
if minDest is dest: return minPath
if minDest is in finished: continue
for each edge e = ⟨minDest, child⟩:
// If we don’t know the minimum-cost path from
start to child, // examine the path we've just found
if child is not in finished: newPath = minPath + e add newPath to active
add minDest to finished
If the loop terminates, then no path exists from start to dest.
The implementation should indicate this to the client.
Dijkstra’s Algorithm in Marvel Paths
You will write a modified version of your Marvel Paths application in which your application finds its paths using Dijkstra’s algorithm instead of
BFS. Dijkstra’s algorithm requires weighted edges. To simulate edge weights over the Marvel dataset, the weight of the edge between two characters will be based on how well-connected those two characters are. Specifically, the weight is the inverse of how many comic books two characters are in together (equivalently, the weight is the multiplicative inverse of the number of edges in the multigraph between them). For example, if Amazing Amoeba and Zany Zebra appeared in 5 comic books together, the weight of the edge between them would be 1/5. Thus, the more well-connected two characters are, the lower the weight and the more likely that a path is taken through them. In summary, the idea with the Marvel data is to treat the number of paths from one node to another as a “distance” — if there are several edges from one node to another then that is a “shorter” distance than another pair of nodes that are only connected by a single edge.
Things to note:
- A path from a character to itself is defined to have cost 0.0.
- Calculations for the weight of the edges in your graph should be
done when the graph is loaded. This assignment is different from the last one in that when the graph is initialized there is only one edge between nodes and that edge is the weighted edge. The one edge between any two characters will have the label containing the multiplicative inverse of how many books they share.
- On graph initialization, there will be at most one edge between two Marvel characters, but the addEdge functionality still allows adding multiple weighted edges between two characters. You must be sure that path-finding will use the least cost edge if there are multiple edges between two characters. In other words, if a new edge is added to your graph after its initialization, then path-finding should be able to factor that edge into its calculations without re-calculating the weights of other edges or deleting/overwriting any existing edges.
Place your new Marvel application in src/main/java/hw6/ MarvelPaths2.java in package hw6. In choosing how to organize your code, remember to avoid duplicating code as much as possible. In particular, reuse existing code where possible, and keep in mind that you may need to use the same implementation of Dijkstra’s algorithm in a different application.
For this assignment, your program must be able to construct the graph and find a path in less than 30 seconds using the full Marvel dataset. We will
set the tests to have a 3000 ms (30 second) timeout for each test and run with assertions disabled.
As before, you are welcome to write a main() method for your application, but you are not required to do so.
The interface of MarvelPaths2 is the same as that of MarvelPaths in Homework 5, but with a few differences in arguments and output:
- public void createNewGraph(String filename) is the same. It creates a brand new graph and populates the graph from filename, where filename is a data file of the format for marvel.csv and is located in data/. As in Homework 5, relative paths to data files should begin at data/. Consult Section “Paths to files” in Homework 5 if you are having trouble making Eclipse work with these relative paths.
- public String findPath(String CHAR1, String CHAR2) searches with Dijkstra’s algorithm instead of BFS and returns its output in the form: path from CHAR1 to CHARN:
- CHAR1 to CHAR2 with weight w1
- CHAR2 to CHAR3 with weight w2
- CHARN-1 to CHARN with weight wN-1
For readability, the output of findPath should print numeric values with exactly 3 digits after the decimal point, rounding to the nearest value if they have more digits. The easiest way to specify the desired format of a value is using format strings. For example, you could create the String “Weight of 1.760” by writing: String.format(“Weight of %.3f”, 1.759555555555);
In findPath, the total cost should be computed by summing the full values of the individual weights, not the rounded values.
- As in Homework 5, a path from a character to itself should be treated as a trivially empty path. Because this path contains no edges, it has a cost of zero. (Think of the path as a list of edges. The sum of an empty list is conventionally defined to be zero.) So, your findPath should produce the usual output for a path but without any edges, i.e.:
path from C to C:
- total cost: 0.000
This only applies to characters in the dataset: a request for a path from a character that is not in the dataset to itself should print the usual “unknown character C” output.
- Also as in Homework 5, if the user gives two valid node arguments CHAR1 and CHAR2 that have no path in the specified graph, output:
path from CHAR1 to CHARN:
no path found
Problem 3: Testing Your Solution
Just as with Homework 5, create smaller *.csv files to test your graph building and findPaths. Place them in the data/ directory. Write tests in JUnit classes and place those tests in hw6 package (src/test/java/hw6/ directory). Just as in Homework 4 and 5, run EclEmma to measure your code coverage. The code coverage threshold will be set at 85% for this assignment.
Tests do not directly test the property that your graph is generic. However, Homework 4 and Homework 5 tests use String edge labels, while Homework 6 uses numeric values. Supporting all three test drivers implicitly tests the generic behavior of your graph.
Reflection [0.5 points]
Please answer the following questions in a file named hw6_reflection.pdf in your answers/ directory. Answer briefly, but in enough detail to help you improve your own practice via introspection and to enable the course staff to improve Principles of Software in the
future. .txt files will also be accepted.
- In retrospect, what could you have done better to reduce the time you spent solving this assignment?
- What could the course staff have done better to improve your learning experience in this assignment?
- What do you know now that you wish you had known before beginning the assignment?
Collaboration [0.5 points]
Please answer the following questions in a file named hw6_collaboration.pdf in your answers/ directory. .txt files will also be accepted.
The standard integrity policy applies to this assignment.
State whether or not you collaborated with other students. If you did collaborate with other students, state their names and a brief description of how you collaborated.
- Instructor hw4 tests: 5 pts (auto-graded)
- Instructor hw5 tests: 5 pts (auto-graded)
- Quality of hw6 test suite, percent of your tests passed: 5 pts (auto-
- Quality of hw6 test suite, percent coverage: 5 pts (auto-graded)
- Instructor MarvelPaths2 tests: 16 pts (auto-graded)
- Code quality (Code organization, Genericity, Rep invarians,
Abstraction functions, Specifications): 13 pts
- Collaboration and reflection: 1 pt
src/main/java/hw6/MarvelPaths2.java src/main/java/hw6/*.java [Other classes you create, if any] data/*.csv [Your .csv test files. Do not commit marvel.csv]
src/test/java/hw6/*Test.java [JUnit test classes you create] answers/hw6_reflection.pdf answers/hw6_collaboration.pdf
Additionally, be sure to commit any updates you make to the following files:
- src/test/java/hw5/*Test.java [Your Marvel Paths test classes]
None yet. Check the Submitty Discussion Form page regularly. As usual, we will announce errors there.
Parts of this homework are copied from Mike Enrst’s Software Design and Implementation course.
本网站支持淘宝 支付宝 微信支付 paypal等等交易。如果不放心可以用淘宝交易！
E-mail: [email protected] 微信:itcsdx