Thanh's Islet 🏝️

Leetcode 6: Zigzag Conversion

Description

The original description is available here. This is my take on it however:

We put characters of string s into a grid with row count n following a zigzag pattern, then return a “result” string, which is the characters themselves joined left to right, up to down.

For example

# with
s = "abcdefghijklmno"
n = 3

# then
result = "aeimbdfhjlncgko"

For clarification, putting the string on a grid for zigzag is like this:

+-+-+-+-+-+-+-+
|a| |e| |i| |m|
+-+-+-+-+-+-+-+
|b|d|f|h|j|l|n|
+-+-+-+-+-+-+-+
|c| |g| |k| |o|
+-+-+-+-+-+-+-+

Heart of The Problem

|  /|  /|
| / | / |
|/  |/  |
1.               2.               3.               4.             
+-+-+-+-+-+-+-+  +-+-+-+-+-+-+-+  +-+-+-+-+-+-+-+  +-+-+-+-+-+-+-+
|*| | | | | | |  |*| | | | | | |  |*| | | | | | |  |*| | | | | | |
+-+-+-+-+-+-+-+  +-+-+-+-+-+-+-+  +-+-+-+-+-+-+-+  +-+-+-+-+-+-+-+
| | | | | | | |  |*| | | | | | |  |*| | | | | | |  |*|*| | | | | |
+-+-+-+-+-+-+-+  +-+-+-+-+-+-+-+  +-+-+-+-+-+-+-+  +-+-+-+-+-+-+-+
| | | | | | | |  | | | | | | | |  |*| | | | | | |  |*| | | | | | |
+-+-+-+-+-+-+-+  +-+-+-+-+-+-+-+  +-+-+-+-+-+-+-+  +-+-+-+-+-+-+-+

5.               6.               7.               8.             
+-+-+-+-+-+-+-+  +-+-+-+-+-+-+-+  +-+-+-+-+-+-+-+  +-+-+-+-+-+-+-+
|*| |*| | | | |  |*| |*| | | | |  |*| |*| | | | |  |*| |*| |…| | |
+-+-+-+-+-+-+-+  +-+-+-+-+-+-+-+  +-+-+-+-+-+-+-+  +-+-+-+-+-+-+-+
|*|*| | | | | |  |*|*|*| | | | |  |*|*|*| | | | |  |*|*|*|*| | | |
+-+-+-+-+-+-+-+  +-+-+-+-+-+-+-+  +-+-+-+-+-+-+-+  +-+-+-+-+-+-+-+
|*| | | | | | |  |*| | | | | | |  |*| |*| | | | |  |*| |*| | | | |
+-+-+-+-+-+-+-+  +-+-+-+-+-+-+-+  +-+-+-+-+-+-+-+  +-+-+-+-+-+-+-+
1.               2.               3.               4.             
+-+-+-+-+-+-+-+  +-+-+-+-+-+-+-+  +-+-+-+-+-+-+-+  +-+-+-+-+-+-+-+
|a| | | | | | |  |a| | | | | | |  |a| | | | | | |  |a| | | | | | |
+-+-+-+-+-+-+-+  +-+-+-+-+-+-+-+  +-+-+-+-+-+-+-+  +-+-+-+-+-+-+-+
| | | | | | | |  |b| | | | | | |  |b| | | | | | |  |b|d| | | | | |
+-+-+-+-+-+-+-+  +-+-+-+-+-+-+-+  +-+-+-+-+-+-+-+  +-+-+-+-+-+-+-+
| | | | | | | |  | | | | | | | |  |c| | | | | | |  |c| | | | | | |
+-+-+-+-+-+-+-+  +-+-+-+-+-+-+-+  +-+-+-+-+-+-+-+  +-+-+-+-+-+-+-+

5.               6.               7.               8.             
+-+-+-+-+-+-+-+  +-+-+-+-+-+-+-+  +-+-+-+-+-+-+-+  +-+-+-+-+-+-+-+
|a| |e| | | | |  |a| |e| | | | |  |a| |e| | | | |  |a| |e| |…| | |
+-+-+-+-+-+-+-+  +-+-+-+-+-+-+-+  +-+-+-+-+-+-+-+  +-+-+-+-+-+-+-+
|b|d| | | | | |  |b|d|f| | | | |  |b|d|f| | | | |  |b|d|f|h| | | |
+-+-+-+-+-+-+-+  +-+-+-+-+-+-+-+  +-+-+-+-+-+-+-+  +-+-+-+-+-+-+-+
|c| | | | | | |  |c| | | | | | |  |c| |g| | | | |  |c| |g| | | | |
+-+-+-+-+-+-+-+  +-+-+-+-+-+-+-+  +-+-+-+-+-+-+-+  +-+-+-+-+-+-+-+
+-+-+-+-+-+-+-+
|a| |e| |i| |m|
+-+-+-+-+-+-+-+
|b|d|f|h|j|l|n|
+-+-+-+-+-+-+-+
|c| |g| |k| |o|
+-+-+-+-+-+-+-+
row_0 = ['a', 'e', 'i', 'm']
row_1 = ['b', 'd', 'f', 'h', 'j', 'l', 'n']
row_2 = ['c', 'g', 'k', 'o']
|  / |  / |  /
| /  | /  | /
|/   |/   |/

1.   2.   3.   ...
+-+-+  +-+-+  +-+-+
|a| |  |e| |  |i| |
+-+-+  +-+-+  +-+-+
|b|d|  |f|h|  |j|l|  ...
+-+-+  +-+-+  +-+-+
|c| |  |g| |  |k| |
+-+-+  +-+-+  +-+-+
   0 1    2 3    4 5
  +-+-+  +-+-+  +-+-+
0 |a| |  |e| |  |i| |
  +-+-+  +-+-+  +-+-+
1 |b|d|  |f|h|  |j|l|  ...
  +-+-+  +-+-+  +-+-+
2 |c| |  |g| |  |k| |
  +-+-+  +-+-+  +-+-+

- a: (0, 0)
- b: (1, 0)
- c: (2, 0)
- d: (1, 1)

- e: (0, 2)
- f: (1, 2)
- g: (2, 2)
- h: (1, 3)

- ...
- a: (0, 0)
     +1
- b: (1, 0)
     +1
- c: (2, 0)
     -1 +1
- d: (1, 1)
     -1 +1
- e: (0, 2)
     +1
- f: (1, 2)
     +1
- g: (2, 2)
     -1 +1
- h: (1, 3)
     -1 +1
...

Implementation

Seeing the changes as an infinite stream, we can leverage Python’s iterator like this:

def generate_changes(num_rows: int) -> Iterator:
    while True:
        # step down
        yield from repeat((1, 0), num_rows - 1)
        # step up right
        yield from repeat((-1, 1), num_rows - 1)

Then generating the positions is a bit more tricky:

def generate_positions(
    start_position: (int, int),
    num_rows: int,
) -> Iterator:
    changes = generate_changes(num_rows=num_rows)
    current_position = start_position
    while True:
        yield current_position
        change = next(changes)
        x, y = current_position
        change_x, change_y = change
        current_position = (x + change_x, y + change_y)

We wrap it up in the class Solution that LeetCode provides us:

class Solution:
    def convert(self, s: str, numRows: int) -> str:
        if numRows == 1:
            return s
        positions = generate_positions(start_position=(0, 0), num_rows=numRows)
        s_with_position = list(zip(s, positions))
        s_sorted_by_position = sorted(s_with_position, key=lambda pair: pair[1])
        s_sorted_without_position = list(map(lambda pair: pair[0], s_sorted_by_position))
        result = ''.join(s_sorted_without_position)
        return result

Conclusion

Some people may not like the original description of the problem, as it is too vague and lack of explanation, but I found the decrypting process fun anyway.

Also, the solution that I proposed might not be the most optimal one, but it is intuitive for me anyway.

#leetcode #python