Skip to content

Latest commit

 

History

History
 
 

day11

Folders and files

NameName
Last commit message
Last commit date

parent directory

..
 
 
 
 

question of the day: print the contents of an n by m matrix in a spiral.

For example:

input:

[[1,  2,  3,  4],
[ 5,  6,  7,  8],
[ 9,  10, 11, 12],
[ 13, 14, 15, 16]]

output:

1 2 3 4 8 12 16 15 14 13 9 5 6 7 11 10

Ideas

(quick preface):
I've always been mildly allergic to these kinds of matrix and array manipulation problems. Off-by-one errors, translating visual movements into for-loop and while-loop logic blocks, etc. Therefore doing more of these problems will help me conquer that weakness. And I think just doing one of these per day is not enough, especially if I'm skipping from topic to topic. What I'll do is clump a bunch of these kinds of questions together, and do them as a series day to day so that over time I can get a sense of some patterns. Pattern recognition is what's going to be truly useful to me.

(actual ideas about the problem):
Edge cases (?) include empty matrix, having only one row to print, having only one column to print, having a row width vastly different from the column height.

Let's just assume we'll be printing in a clockwise fashion.

Here are some sample inputs and outputs:

inputs outputs
[[]]
[[1,2]] 1 2
[[1],
[2],
[3],
[4]]
1 2 3 4
[[8, 1],
[7, 2],
[6, 3],
[5, 4]]
1 2 3 4 5 6 7 8

Alright now let's get a brute force solution down and then see if it's possible to optimize.

If I wanted to print row by row, that'd be easy: two nested for-loops one to go through the rows, and one to go through each of the elements in each of the rows. The trick with printing in a spiral is that sometimes we want to go up, sometimes we want to go down in terms of the index of the elements. So now I'm thinking we need to have some cursor comprised of a row index row and column index col, and we also need to be aware of the current direction we're iterating in so that it's possible to change directions.

Here's the strategy:

Start at (0,0) with direction right. Keep adding to the row index until we hit the end of the row. Switch the direction to the next direction, down. Keep adding to the column index until we hit the end of the column. Switch the direction to the next direction, left. Keep subtracting from the row index until we hit the beginning of the row. Switch the direction to the next direction, up. Keep subtracting from the column index until we hit the beginning of the column.

Every time we finish a row, we have one less column index to go through, and vice versa. Repeat this cycle until all the elements in the matrix have been printed, and watch out for the off-by-ones when checking for the boundaries of the matrix.

We need boundary variables that will tell us when it's necessary to change directions.

We'll have a big while loop and four while loops inside the big one. The big while loop breaks only when there are no possible moves to make within the current boundaries. The four while loops inside loop through the four different directions.

Aiight let's do this:

Code

Python