SPOJ Problem Set (classical) - Fishmonger

dynamic programming graph Floyd Warshall spoj

Problem:

Please find the problem here.

Solution:

I started the problem using the a simple complete search of all path. Of course, we are not going to be able to enumerate through all the paths. But fortunately we don’t have to, once we have tried a path we can memoize the result as follow.

If I am at a certain node n1 with certain an initial time constraint T1before, and with time t12 I can get to node n2, recursively we know the best path to get to market from n2 with budget T-t12 take time T2after and has minimal cost C2after, then we have one best path candidate from n1 to n2 within time constraint T1before that actually spend time T2after + t12 with cost C2after + c12. Try all candidate and find the solution.

The key to the above idea is memoization - we save the fact that if we are going from node n1, the budget given is between T1before and T2after + t12, then the best path has the be the best path found, no point to find a best path again.

I have coded the solution, unforunately it is Wrong Answer. Removed memoization, clearly Time Limit Exceeded.  Looking in retrospect, could be similar bug that I hit in the approach below.

Now I tried something else, based on Floyd Warshall. Floyd Warshall is that we can find the shortest path between every two nodes by allowing incrementally more intermediate nodes. I thought, if instead of the shortest path, what if we maintain the set of non dominated paths? By non-dominated, I mean paths that has either lower cost higher time, or path with higher time lower cost, but not higher time, higher cost ones. We also constraint the time of those path within time budget. At the end of the algorithm, we will get, between all pairs of nodes, the set of all non-dominated paths, and therefore we can pick the one with least cost within time budget at the end of the algorithm.

For non-dominated filtering, I just used the simple \( O(p^2) \) algorithm, where \( p \) is the number of candidate paths. There exists \( O(p \log p) \) algorithm, and maybe even better, I just didn’t do it since the time is good enough for the problem to accept.

All is fine, except one bug I just can’t find myself. When I initialized the table, I put the initial distance and initial cost between two nodes into the table. I didn’t check if those values are within time budget, as a result, if there is a graph with a low cost path within short time budget that reach the market directly, I would have returned that path - wrong.

If I got more time, I can try modifying Dijkstra’s algorithm or Bellman Ford’s algorithm, basically the same idea of keeping non-dominated paths. These single source shortest path algorithm should be faster than the all pair shortest path problem, I believe.

As an aside - thanks a lot for those who helped me to find out the tricky test case that I missed. I guess I need to be a better adversary of my own code to find that out. I also got the feedback that the code is hard to read, I tried to make my code as descriptive as it can be, but I guess everybody have a different taste of what does it mean by readable, or well, code are just not fun to read, wrapping it with sugar just doesn’t cover the bitterness.

Code: