Level
프로그래머스 Lv3
 
 
function solution(board) {

    const minCost = board.map((row) => row.map((i) => Infinity));
    minCost[0][0] = 0;

    const isSafe = (x,y) => !board[y] ? false : board[y][x]===0 ? true : false;

    const stack = [];
    const start = [[1,0],[0,1]];
    start.forEach((point,i) => isSafe(...point) && stack.push([...point, i, 100]));

    while(stack.length){
				const vertex = stack.shift();
        const [x,y,dir,cost] = vertex;

        if(cost <= minCost[y][x]){
            minCost[y][x] = cost;

            const right = [x+1, y, 0 ];
            const left = [x-1, y, 0 ];
            const top = [x, y-1, 1 ];
            const bottom = [x,y+1, 1 ];
            
            if(dir){
                isSafe(...top) && stack.push([ ...top, cost+100]);
                isSafe(...bottom) && stack.push([ ...bottom, cost+100]);
                isSafe(...left) && stack.push([ ...left, cost+600]);
                isSafe(...right) && stack.push([ ...right,cost+600]);   
            }else{
                isSafe(...left) && stack.push([...left, cost+100]);
                isSafe(...right) && stack.push([...right, cost+100]);
                isSafe(...top) && stack.push([...top, cost+600]);
                isSafe(...bottom) && stack.push([ ...bottom, cost+600]);
            }
        }
    }
    return minCost[board.length-1][board.length-1];
}
 
1) 100, 600, 100, 600 순으로 입력될 경우
function solution(board) {
  const isSafe = (x, y) =>
    !board[y] ? false : board[y][x] === 0 ? true : false;

  const memo = board.map((row) => row.map((i) => Infinity));
  memo[0][0] = 0;

  const moves = [1, 1, -1, -1]; // 우 하 좌 상

  const stack = [];

  [[1,0],[0,1]].forEach(
    (point, dir) => isSafe(...point) && stack.push([dir, ...point, 100])
		// dir이 1이면 수직, 0이면 수평 방향
  );

  while (stack.length) {
    const [dir, x, y, cost] = stack.shift();

    if (cost <= memo[y][x]) {
      memo[y][x] = cost;

      moves.forEach((move, i) => {
        const nextDir= i % 2; // 1이면 수직, 0이면 수평
        const nextPoint = nextDir ? [x, y + move] : [x + move, y];

        isSafe(...newPoint) && 
            (nextDir===dir 
            ? stack.push([nextDir, ...nextPoint , cost+100])
            : stack.push([nextDir, ...nextPoint, cost+600]));
      });
    }

  }

  return memo[board.length - 1][board.length - 1];
}
 
2) 최소 charge 부터 넣어줄 경우
function solution(board) {
  const isSafe = (x, y) =>
    !board[y] ? false : board[y][x] === 0 ? true : false;

  const memo = board.map((row) => row.map((i) => Infinity));
  memo[0][0] = 0;

  const moves = [1, 1, -1, -1]; // 우 하 좌 상

  const stack = [];

  [[1,0],[0,1]].forEach(
    (point, dir) => isSafe(...point) && stack.push([dir, ...point, 100])
  );

  while (stack.length) {
    const [dir, x, y, cost] = stack.shift();

    const minCharge = [];
    const maxCharge = [];

    if (cost <= memo[y][x]) {
      memo[y][x] = cost;

      moves.forEach((move, i) => {
        const nextDir = i % 2;
        const nextPoint = nextDir ? [x, y + move] : [x + move, y];

        isSafe(...nextPoint) && 
            (dir===nextDir 
             ? minCharge.push([nextDir, ...nextPoint, cost + 100]) 
             : maxCharge.push([nextDir, ...nextPoint, cost + 600]));
      });
    }

    stack.push(...minCharge, ...maxCharge);
  }

  return memo[board.length - 1][board.length - 1];
}
 
3) 방향을 추가한 메모아이징 , dx/dy 사용해 리팩토링
 
function solution(board) {

    const minCost = board.map((row) => row.map((i) => [Infinity, Infinity]));
    minCost[0][0] = [0,0];

    const isSafe = (x,y) => !board[y] ? false : board[y][x]===0 ? true : false;

    const dxdy = [
        [[1,-1,0,0],[0,0,1,-1]],
        [[0,0,1,-1],[1,-1,0,0]]
    ];
    
    const stack = [];
    [[1,0],[0,1]].forEach((start,i) => isSafe(...start) && stack.push([...start, i,i, 100]));

    while(stack.length){
        const [x,y,startDir,dir,cost] = stack.shift();

        if(cost <= minCost[y][x][startDir]){
            minCost[y][x][startDir] = cost;
            
            for(let i=0; i<4; i++){
                const xx = x + dxdy[dir][0][i];
                const yy = y + dxdy[dir][1][i];
                isSafe(xx, yy) && stack.push([xx, yy, startDir, xx === x ? 1 : 0, cost + (i<2 ? 100 : 600)]);
            }
        }
    }
    return Math.min(...minCost[board.length-1][board.length-1]);
}
 
예외 테스트케이스 통과
int[][] board = {
        {0, 0, 0, 0, 0},
        {0, 1, 1, 1, 0},
        {0, 0, 1, 0, 0},
        {1, 0, 0, 0, 1},
        {0, 1, 1, 0, 0}
    };
[[0,0,0,0,0],[0,1,1,1,0],[0,0,1,0,0],[1,0,0,0,1],[0,1,1,0,0]];
 
4) 이동 방향이 반대 방향인지 확인
function solution(board) {

    const minCost = board.map((row) => row.map((i) => [Infinity, Infinity]));
    minCost[0][0] = [0,0];

    const isSafe = (x,y) => !board[y] ? false : board[y][x]===0 ? true : false;

    const dxdy = [
        [[1,-1,0,0],[0,0,1,-1]],
        [[0,0,1,-1],[1,-1,0,0]]
    ];
    
    const stack = [];
    [[1,0],[0,1]].forEach((start,i) => isSafe(...start) && stack.push([...start, i ,i, 100]));

    while(stack.length){
        const [x,y,startDir,dir,cost] = stack.shift();

        if(cost < minCost[y][x][startDir]){
            minCost[y][x][startDir] = cost;
            
            for(let i=0; i<4; i++){
                const dx = dxdy[Math.abs(dir)][0][i];
                const dy = dxdy[Math.abs(dir)][1][i];
                const xx = x + dx;
                const yy = y + dy;
                const nextDir = dx > 0 ? 0 : dx < 0 ? '0' : dy;
                if(dir!==nextDir && dir + nextDir==0){ // 반대방향 이동
                    continue;
                }
                
                isSafe(xx, yy) && stack.push([xx, yy, startDir, nextDir, cost + (i<2 ? 100 : 600)]);
            }
        }
    }
    return Math.min(...minCost[board.length-1][board.length-1]);
}