프로그래밍/알고리즘

프로그래머스/위클리 챌린지/퍼즐 조각 채우기

Doublsb 2021. 8. 18. 02:31

https://programmers.co.kr/learn/courses/30/lessons/84021

 

풀이

- 보드 빈칸에 회전시킬 수 있는 조각을 채워넣어야 한다

- 최대한 많은 조각을 채워넣고, 몇 칸을 채웠는지 출력하면 된다

 

- 보드에 어떤 위치에 조각이 들어있는지는 상관 없다. 모양이 맞기만 하면 되는거라 "빈 칸"과 "조각들"만 구분하면 된다.

- 해당 빈 칸과 같은 모양의 조각이 있는지 회전해가며 찾으면 된다.

- 어려운 부분은 "조각을 골라내기"이다.


1) 조각을 골라내는 방식 정하기



조각을 떼어내려면 각 칸의 연결성을 알아내야 한다.

나는 맨 첫번째 칸부터 시작해서 우측으로 이동하며 연결성을 알아보기로 했다.

이 경우 좌측, 상단의 칸만 조사하면 되기 때문이었다.

다만, 우측으로 연결되었음에도 불구하고 좌측과 상단이 빈 경우가 있었다.

 

이 경우는 일단 분리된 조각으로 판단했다.

그 후, 좌측과 상단을 조사했을 때 서로 다른 블록을 가리키는 경우가 있다면 해당 블록들을 통합했다.

어느 좌표가 어떤 블록에 포함되어있는지 빠른 검색을 위해 Dictionary를 이용했고,

블록 객체에서는 해당 객체에 포함된 좌표들을 해시셋으로 보관했다.


2) 조각을 int[,] 데이터로 변환하고, 회전하면서 맞추기

조각을 골라냈다면 이후는 간단하다. 2차원 배열을 회전시키면서 맞추기만 하면 되기 때문이다.

참고로 테이블에 있던 조각들 중에서 이미 끼워 넣은 것들을 제외시키지 않으면 틀리게 될 것이다 (...) 경험담.

 

 

C#

using System;
using System.Linq;
using System.Collections.Generic;

public class Solution {
    public int solution(int[,] game_board, int[,] table) {
        
        int answer = 0;
        
        var GameBoard = new board(game_board, 0).Get_Blocks();
        var Table = new board(table, 1).Get_Blocks();

        foreach(var board_block in GameBoard) {
            
            int[,] finded = null;
            
            foreach(var table_block in Table) {
                
                if(isEqual(board_block, table_block)) {
                    finded = table_block;
                    break;
                }
                
                else {
                    int[,] rotated = table_block;
                    
                    for(int i=0; i<3; i++) {
                        rotated = Rotate90(rotated);
                        if(isEqual(board_block, rotated)) {
                            finded = table_block;
                            break;
                        }
                    }
                }
            }
            
            if(finded != null) {
                answer 
                += board_block.Cast<int>().Count(e => e == 1);
                
                Table.Remove(finded);
            }
        }
        
        return answer;
    }
    
    public int[,] Rotate90(int[,] data) {
        
        int width = data.GetLength(1);
        int height = data.GetLength(0);
        int[,] result = new int[width, height];

        for(int i=0; i<width; i++){
            for(int j=0; j<height; j++){
                result[i, j] = data[j, width-i-1];
            }
        }
        
        return result;
    }

    private bool isEqual(int[,] A, int[,] B) {
        bool result = true;
        
        if(A.GetLength(0) != B.GetLength(0) 
           || A.GetLength(1) != B.GetLength(1)) return false;
        
        for(int i=0; i<A.GetLength(0); i++){
            for(int j=0; j<A.GetLength(1); j++){
                if(A[i, j] != B[i, j]){
                    result = false;
                    break;
                }
            }
        }
        
        return result;
    }
}


public class board {

    public Dictionary<Vector2, Block> Map;
    
    public board(int[,] data, int validValue) { 
        Map = new Dictionary<Vector2, Block>();
        Initialize(data, validValue);
    }
    
    public void Initialize(int[,] data, int validValue) {
        
        int size = data.GetLength(0);

        for(int i=0; i<size; i++){
            for(int j=0; j<size; j++){
                
                Vector2 key = new Vector2(i, j);
                
                if(data[key.x, key.y] == validValue && !Map.ContainsKey(key))
                    connect(getConnectedBlock(key), key);
            }
        } 
    }
    
    public List<int[,]> Get_Blocks() {
        
        var result = new List<int[,]>();
        
        foreach(var block in Map.Values.Distinct()) {
            result.Add(block.ToArray());
        }
        
        return result; 
    }
    
    private void connect(Block block, Vector2 key){
        block.Add(key);
        Map.Add(key, block);
    }
    
    private Block getConnectedBlock(Vector2 key) {
        var leftKey = new Vector2(key.x, key.y - 1);
        var upKey = new Vector2(key.x - 1, key.y);
        
        var left = Map.ContainsKey(leftKey) ? Map[leftKey] : null;
        var up = Map.ContainsKey(upKey) ? Map[upKey] : null;

        if(left != up && left != null && up != null) combine(left, up);
        
        if(left != null) return left;
        else if(up != null) return up;
        else return new Block();
    }
    
    private void combine(Block a, Block b){
        foreach(var e in b.List) { 
            a.List.Add(e);
            Map[e] = a;                    
        }
    }
}

public class Block {
    public HashSet<Vector2> List 
        = new HashSet<Vector2>();
    
    public void Add(Vector2 key) {
        List.Add(key);
    }
    
    public int[,] ToArray() {
        var pivotX = List.Min(e => e.y);
        var pivotY = List.Min(e => e.x);

        var width = List.Max(e => e.y) - pivotX + 1;
        var height = List.Max(e => e.x) - pivotY + 1;

        var result = new int[height, width];

        foreach(var e in List)
            result[e.x - pivotY, e.y - pivotX] = 1;

        return result;
    }
}

public struct Vector2 {
    public int x, y;
    public Vector2(int x, int y) { this.x = x; this.y = y;}
}

 

결과

100.0/100.0

피드백

해결하는데 너무 오래 걸렸다. 일단 조각을 골라낼 때 삽질을 너무 많이 함.

하지만 내가 C#으로 제일 처음 해결한 사람이라서 매우 뿌듯함. 젠장! 이 맛에 하는 거로군!

 

그리고 더러운 코드라고 느껴서 리팩터링을 시도(...)했었는데, 제일 처음으로 정답이었던 코드만 기록되서 과감히 버렸다.

다음엔 코딩테스트 문제는 풀고 나서 리팩터링하지 말아야겠다. 그럴 시간에 내 프로젝트 코드나 리팩터링하라고 ^w^...

반응형