algorithms data structures optimization graph algorithms greedy complexity dynamic programming python