6. Zigzag Conversion
Problem Description
The task is to rearrange the characters of a given string s
in a zigzag pattern on a specified number of rows and then read off the characters line by line to create a new string. The zigzag pattern means that the characters of s
are placed diagonally in a zigzag manner in downwards and upwards directions successively. After placing all the characters in this pattern, we would read the characters in each row horizontally and concatenate to form the final string.
To visualize the process, you can think of writing down the characters in the following way:
- Starting from the top row, write characters downwards.
- Upon reaching the last row, switch direction and start writing characters upwards, forming a diagonal line until you reach the top row again.
- Alternate the process until every character is written in this zigzag fashion.
For example, given a string "PAYPALISHIRING"
and 3 rows, the characters should be placed like this:
1P A H N 2A P L S I I G 3Y I R
After arranging the characters, the string "PAHNAPLSIIGYIR"
is obtained by reading each row sequentially. To form this new string programmatically, the code should simulate the reading process.
Intuition
The intuitive approach to simulate the zigzag conversion process involves creating an imaginary 2D grid that represents rows and columns of the pattern. A group
is defined which denotes the cycle length of the zigzag pattern, calculated as 2 * numRows - 2
. If numRows
is 1, then the zigzag pattern is not possible, and the original string is returned.
The solution uses a loop to construct the new string by considering each row individually, assembling the characters that would appear in that row in the zigzag pattern. The interval
between characters of the same row varies depending on their position (at the top or bottom, itโs equal to one cycle; in between, it alternates between moving vertically down and obliquely up) and is used to calculate the index of the next character to be added.
- Initialize an empty list
ans
to hold characters in the order they should be added to form the zigzag string. - Loop through each row. For the top and bottom rows, characters occur after every full cycle (the
group
), so the interval is constant. For intermediate rows, the interval alternates. We calculate the initialinterval
based on the current row. - Within each row, loop to add characters to
ans
, incrementing the index by the interval each time. After each addition, update the interval to alternate between moving vertically and diagonally, accounting for edge cases to avoid a zero interval. - After processing all rows, join the list of characters
ans
into a string to get the final result.
The core of the solution relies on finding the pattern of intervals between the positions within the zigzag alignment and using it to construct the final string efficiently without simulating the entire grid.
Solution Approach
The implementation for the given problem follows a direct approach that calculates the next index in the original string for each character in the transformed zigzag string. It does this by determining the appropriate index intervals without needing to simulate the entire 2D zigzag structure.
Here's how the implementation works:
-
Early Return for Single Row: If
numRows
is 1, the zigzag pattern is trivial and the original string can be returned immediately, as there is no alteration in the order of characters.1if numRows == 1: 2 return s
-
Calculating the Group Cycle: The group cycle (
group
) is calculated as the number of characters that form the complete vertical and diagonal cycle in the zigzag pattern, given by2 * numRows - 2
. This number is central to the implementation as it helps to understand the symmetry in the pattern, which repeats everygroup
characters.1group = 2 * numRows - 2
-
Initializing the Answer List: An empty list
ans
is initialized to store the characters in their new order. -
Populating Each Row: The algorithm iterates over each row and identifies the characters that would appear in that row in a zigzag pattern:
- Determine the initial
interval
for characters in the current row. For the first and last rows, it remains constant and equal togroup
. For intermediate rows, theinterval
alternates between two values. - Use a
while
loop to continue adding characters to theans
list until the end of the string is reached.
1for i in range(1, numRows + 1): 2 interval = group if i == numRows else 2 * numRows - 2 * i 3 idx = i - 1 4 while idx < len(s): 5 ans.append(s[idx]) 6 idx += interval 7 interval = group - interval if interval != group else interval
Here,
i
is the row index,idx
is the index in the original string for the current character, and theinterval
determines the number of steps to the next character in the current row.- For intermediate rows (rows between the first and last), after each character is added, the interval for the next character is updated to the complementary interval (the difference between the
group
and the current interval), thus simulating the zigzag pattern without explicitly constructing a 2D grid structure.
- Determine the initial
-
Concatenating the Results: After the rows are processed and all characters are placed in the
ans
list, the list is joined into a string to form the final zigzag transformed string.1return ''.join(ans)
This implementation optimizes space by avoiding the creation of a 2D grid and computes the string in O(n) time by analyzing the inherent pattern in the positions of characters in the zigzag arrangement.
Ready to land your dream job?
Unlock your dream job with a 2-minute evaluator for a personalized learning plan!
Start EvaluatorExample Walkthrough
Let's consider a small example to illustrate the solution approach. Suppose we have the string "HELLO"
and the number of rows numRows
is 3. The task is to arrange the characters in a zigzag pattern and then read them line by line.
Step 1: Early Return for Single Row
Since numRows
is not 1, we don't return the string as is, and we proceed with the rest of the algorithm.
Step 2: Calculating the Group Cycle
We calculate the group cycle (group = 2 * numRows - 2
) which in this case is 2 * 3 - 2 = 4
. This means that the vertical and diagonal pattern will repeat every 4 characters.
Step 3: Initializing the Answer List
We initialize an empty ans
list where we will store the characters.
Step 4: Populating Each Row
We iterate over each row and add the relevant characters to the ans
list:
-
For the first row (
i = 1
), the interval will remain constant at 4. The first character is'H'
at index0
. The next character in the zigzag pattern for the first row would be 4 characters away, but since our string is shorter, we stop here for this row. So we add'H'
. -
For the second row (
i = 2
), the interval starts at2 * 3 - 2 * 2 = 2
and will alternate between 2 and (4 - 2) = 2 for subsequent characters. The first character for this row is'E'
at index1
. The next character in the zigzag pattern for the second row is 2 characters away, which is'L'
at index3
. There are no more characters 2 or 4 steps away, so we stop for this row. We add'E'
and'L'
. -
For the third row (
i = 3
), the interval is constant at 4, the same as for the first row. The first character is'L'
at index2
, and there are no more characters 4 steps away. So, we add'L'
.
Step 5: Concatenating the Results
We concatenate the characters in the ans
list to form the final string which is 'HEL'
.
Following the steps above, the zigzag arrangement for the string "HELLO"
with 3 rows would look like this:
1H . 2. E . 3L . . 4. L . 5. . O
When reading off each line, the new string formed is "HEL"
, which matches the characters collected in our answer list.
This walkthrough of a smaller example demonstrates how the designed algorithm would simulate the placement of characters in the zigzag pattern and construct the transformed string accordingly.
Solution Implementation
1class Solution:
2 def convert(self, s: str, num_rows: int) -> str:
3 # If there's only one row or the string is shorter than the number of rows,
4 # it means the pattern will be the same as the input string
5 if num_rows == 1 or num_rows >= len(s):
6 return s
7
8 # Create an array to hold the rows
9 rows = [''] * num_rows
10 # The `step` variable controls the direction the "zigzag" is going.
11 # It starts as 1, meaning "downwards", and will change to -1 to go "upwards".
12 step = -1
13 # Start from the first row
14 curr_row = 0
15
16 # Iterate over each character in the string
17 for char in s:
18 # Append the current character to the current row
19 rows[curr_row] += char
20 # If we're at the top or bottom row, switch direction
21 if curr_row == 0 or curr_row == num_rows - 1:
22 step = -step
23 # Move to the next row in the current direction
24 curr_row += step
25
26 # Concatenate all rows to form the final string
27 return ''.join(rows)
28
29# Example usage:
30# solution = Solution()
31# converted_string = solution.convert("PAYPALISHIRING", 3)
32# print(converted_string) # Output should be "PAHNAPLSIIGYIR"
33
1class Solution {
2 public String convert(String inputString, int numRows) {
3 // If numRows is 1, no pattern is required, so return the string as it is.
4 if (numRows == 1) {
5 return inputString;
6 }
7
8 // StringBuilder is more efficient when appending characters in a loop.
9 StringBuilder convertedStringBuilder = new StringBuilder();
10 // Length of the pattern cycle.
11 int cycleLength = 2 * numRows - 2;
12
13 // Loop over each row.
14 for (int row = 0; row < numRows; row++) {
15 // Calculate the interval for the current row.
16 // For the first and last row, it is the cycle length,
17 // for the others, it depends on the row number.
18 int interval = (row == numRows - 1) ? cycleLength : 2 * (numRows - row - 1);
19 // Index to keep track of the position on the string.
20 int index = row;
21
22 // Continue looping until the end of the string is reached.
23 while (index < inputString.length()) {
24 // Append character at index to the result.
25 convertedStringBuilder.append(inputString.charAt(index));
26 // Proceed to the next character in the current row.
27 index += interval;
28 // Toggle the interval for the middle rows.
29 // This does not affect the first and last rows.
30 interval = (interval == cycleLength || interval == 0) ? cycleLength : cycleLength - interval;
31 }
32 }
33
34 // Convert StringBuilder back to String and return.
35 return convertedStringBuilder.toString();
36 }
37}
38
1#include <string>
2
3class Solution {
4public:
5 /**
6 * Converts the input string to a new string arranged in a zigzag pattern on a given number of rows.
7 *
8 * @param s Input string to be converted.
9 * @param numRows The number of rows in which the string will be rearranged into a zigzag pattern.
10 * @return A string representing the zigzag pattern.
11 */
12 std::string convert(std::string s, int numRows) {
13 // If there is only one row, then the zigzag pattern is the same as the original string.
14 if (numRows == 1) return s;
15
16 std::string ans; // The final answer string in zigzag order.
17 int cycleLength = 2 * numRows - 2; // The length of the repeating zigzag cycle.
18
19 // Loop through each row.
20 for (int currentRow = 1; currentRow <= numRows; ++currentRow) {
21 // The interval depends on the current row and alternates within each zigzag cycle.
22 int interval = (currentRow == numRows) ? cycleLength : 2 * numRows - 2 * currentRow;
23 int currentIndex = currentRow - 1; // The starting index in the original string for this row.
24
25 // Loop through characters in the row.
26 while (currentIndex < s.length()) {
27 ans.push_back(s[currentIndex]); // Append character to the answer string.
28 currentIndex += interval; // Move to the next character in the zigzag pattern.
29 interval = cycleLength - interval; // Alternate the interval for the zigzag pattern.
30
31 // The interval should not be zero; if it is, reset it to the cycle length.
32 if (interval == 0) {
33 interval = cycleLength;
34 }
35 }
36 }
37
38 return ans; // Return the zigzag pattern string.
39 }
40};
41
1function convert(text: string, rowQuantity: number): string {
2 // If the numRows is 1, the "zigzag" pattern is the same as the original string
3 if (rowQuantity === 1) {
4 return text;
5 }
6
7 // Create an array of strings to represent each row initialized with empty strings
8 const rows = new Array(rowQuantity).fill('');
9 let currentRow = 0; // Initialize the current row tracker
10 let goingDown = true; // Flag to determine the direction of iteration through the rows
11
12 // Iterate through each character in the input string
13 for (const char of text) {
14 rows[currentRow] += char; // Append the current character to the current row
15
16 // If we are at the top or bottom row, reverse the direction
17 if (currentRow === 0 || currentRow === rowQuantity - 1) {
18 goingDown = !goingDown;
19 }
20
21 // Move to the next row depending on the direction
22 currentRow += goingDown ? 1 : -1;
23 }
24
25 // Combine all the rows into a single string and return the result
26 return rows.join('');
27}
28
Time and Space Complexity
Time Complexity
The time complexity of the given code is O(n)
. This is determined by the observation that we iterate over each character of the string s
, which has length n
. Every character is visited once during the construction of the ans
list. The nested while loop does not increase the complexity because the inner loop's increments are such that the total number of iterations remains linearly dependent on the length of the s
.
Space Complexity
The space complexity of the code is O(n)
. This is because we use an additional list ans
to store the rearranged characters. In the worst case, the ans
list can contain all characters of the input string s
, which requires space that is linearly proportional to the length of the input string.
Learn more about how to find time and space complexity quickly using problem constraints.
How does merge sort divide the problem into subproblems?
Recommended Readings
Patterns The Shortest Path Algorithm for Coding Interviews The goal of AlgoMonster is to help you get a job in the shortest amount of time possible in a data driven way We compiled datasets of tech interview problems and broke them down by patterns This way we can determine the
Recursion Recursion is one of the most important concepts in computer science Simply speaking recursion is the process of a function calling itself Using a real life analogy imagine a scenario where you invite your friends to lunch https algomonster s3 us east 2 amazonaws com recursion jpg You first
Runtime Overview When learning about algorithms and data structures you'll frequently encounter the term time complexity This concept is fundamental in computer science and offers insights into how long an algorithm takes to complete given a certain input size What is Time Complexity Time complexity represents the amount of time
Got a question?ย Ask the Monster Assistantย anything you don't understand.
Still not clear? ย Submitย the part you don't understand to our editors. Or join ourย Discord and ask the community.