Hello, and welcome back to our coding lesson series. In this unit, we have a fascinating problem at hand that uses the concept of 2D arrays or grids. What's interesting about this problem is that it involves not only simple traversal of the grid but also making this traversal in a unique manner. Initially, the concept might seem a bit tricky, but as we dissect the task and take it step by step, you'll surely enjoy how it unfolds. Are you ready to embark on this adventure? Let's dive right in!
The task before us involves the creation of a TypeScript function named pathTraverse
. This function should perform a particular ordered traversal through a 2D grid. The function will accept a grid represented as a 2D array number[][]
, along with the starting cell coordinates as parameters. Starting from the provided cell, the function should move in any one of the four possible directions toward an adjacent cell. However, a condition governs this selection: the new cell value should be strictly greater than the current cell's value. Think of this as navigating a hiking trail where you can only move to higher ground from your current position.
This pattern would continue as we keep selecting the next available larger cell value. The traversal would halt when there are no cells left that satisfy our criteria. The final result of the function will be an Array<number>
that includes all the visited cell values in the order of their visitation.
Consider a 3x3 grid:
Plain text11 2 3 24 5 6 37 8 9
If we start at the cell with the value 5
, we can logically move to either 6
or 8
. Let's say we choose 8
; the only cell that we can now move to is 9
. After this, we have no more moves left that fit our criteria. Hence, the function returns [5, 8, 9]
.
The first thing we need to do is determine the dimensions of our grid, which is fairly straightforward in TypeScript using the length
property. We can also establish the directions that our traversal can take. In the context of matrices, when we say we are moving "up," we are moving one step toward the first row (decreasing the row index). Similarly, moving "down" corresponds to moving one step toward the last row (increasing the row index), and moving left or right relates to decrementing or incrementing the column index, respectively.
TypeScript1function pathTraverse(grid: number[][], startRow: number, startCol: number): number[] { 2 const rows: number = grid.length; 3 const cols: number = grid[0].length; 4 const directions: [number, number][] = [ 5 [-1, 0], // Up 6 [1, 0], // Down 7 [0, -1], // Left 8 [0, 1] // Right 9 ];
In the directions
array, each pair represents a direction in terms of a pair (rowOffset, colOffset)
. So, if we are at a cell (r, c)
, moving up corresponds to going to the cell (r-1, c)
, moving down corresponds to (r+1, c)
, moving left corresponds to (r, c-1)
, and moving right corresponds to (r, c+1)
.
Once we have the grid's dimensions and the possible directions, we should validate the starting point and set up our visited cell recording mechanism.
TypeScript1function pathTraverse(grid: number[][], startRow: number, startCol: number): number[] { 2 const rows: number = grid.length; 3 const cols: number = grid[0].length; 4 5 // Check the validity of the input 6 if (startRow < 0 || startRow >= rows || startCol < 0 || startCol >= cols) { 7 console.error("Invalid input"); 8 return []; 9 } 10 11 // Define all four possible directions of movement 12 const directions: [number, number][] = [ 13 [1, 0], [-1, 0], [0, -1], [0, 1] 14 ]; 15 16 // Start with the value at the starting cell 17 const visited: number[] = []; 18 visited.push(grid[startRow][startCol]);
The visited.push(grid[startRow][startCol])
line of code appends the value of the current cell to the visited
array, which keeps track of all the cells we've visited so far in the order of their traversal. By doing this, we ensure that our final output will be a sequential list of the values we encounter during our traversal. The visited
array is crucial as it provides the expected result of the function, representing the path taken through the 2D grid.
Let's initiate the grid traversal process in our function. We first set up an infinite loop that will only stop when we break it based on a condition. Inside the infinite loop, for each iteration, we'll have the function try to select the next cell with the maximum value among the adjacent cells. If we find such a cell, we'll capture its value, and it will be our next cell.
TypeScript1function pathTraverse(grid: number[][], startRow: number, startCol: number): number[] { 2 const rows: number = grid.length; 3 const cols: number = grid[0].length; 4 5 // Check the validity of the input 6 if (startRow < 0 || startRow >= rows || startCol < 0 || startCol >= cols) { 7 console.error("Invalid input"); 8 return []; 9 } 10 11 // Define all four possible directions of movement 12 const directions: [number, number][] = [ 13 [1, 0], [-1, 0], [0, -1], [0, 1] 14 ]; 15 16 // Start with the value at the starting cell 17 const visited: number[] = []; 18 visited.push(grid[startRow][startCol]); 19 20 let currMax: number = grid[startRow][startCol]; 21 while (true) { 22 let nextRow: number = -1, nextCol: number = -1; 23 24 // Loop over each adjacent cell in all the directions 25 for (const dir of directions) { 26 // Calculate the new cell's row and column indices 27 const newRow: number = startRow + dir[0]; 28 const newCol: number = startCol + dir[1]; 29 30 // If the new cell is out of the grid boundary, ignore it 31 if (newRow < 0 || newRow >= rows || newCol < 0 || newCol >= cols) { 32 continue; 33 } 34 35 // If the new cell's value is greater than the current maximum 36 if (grid[newRow][newCol] > currMax) { 37 // Save it as the next cell to visit 38 nextRow = newRow; 39 nextCol = newCol; 40 currMax = grid[newRow][newCol]; 41 } 42 } 43 44 // Break the loop if no next cell was found 45 if (nextRow === -1 && nextCol === -1) { 46 break; 47 } 48 49 // Otherwise, go to the next cell 50 startRow = nextRow; 51 startCol = nextCol; 52 53 // Append the cell's value to the result list 54 visited.push(currMax); 55 } 56 57 // Return the list of visited cells' values 58 return visited; 59} 60 61const grid: number[][] = [ 62 [1, 2, 3], 63 [4, 5, 6], 64 [7, 8, 9] 65]; 66const result: number[] = pathTraverse(grid, 1, 1); 67console.log(result); // Output should be [5, 8, 9]
Congratulations! You've successfully solved a complex problem involving the traversal of a grid in a unique pattern using TypeScript. This function has not only challenged your programming skills but also enhanced your understanding of how to utilize TypeScript's type system to catch errors early and improve your coding efficiency.
With this knowledge in hand, it's now time to test your understanding and apply these concepts to similar problems. Look forward to the ensuing practice session, where you can tackle more challenging problems and further refine your problem-solving skills. Keep up the good work and happy coding!