본문으로 바로가기

[BF] 2503번 숫자 야구

category Algorithm/BOJ 문제풀이 2019. 1. 5. 16:03
2503_숫자 야구

2503번 숫자 야구

 

https://www.acmicpc.net/problem/2503


 

문제

정보문화진흥원 정보 영재 동아리에서 동아리 활동을 하던 영수와 민혁이는 쉬는 시간을 틈타 숫자야구 게임을 하기로 했다.

  • 영수는 1에서 9까지의 서로 다른 숫자 세 개로 구성된 세 자리 수를 마음속으로 생각한다. (예: 324)
  • 민혁이는 1에서 9까지의 서로 다른 숫자 세 개로 구성된 세 자리 수를 영수에게 묻는다. (예: 123)
  • 민혁이가 말한 세 자리 수에 있는 숫자들 중 하나가 영수의 세 자리 수의 동일한 자리에 위치하면 스트라이크 한 번으로 센다. 숫자가 영수의 세 자리 수에 있긴 하나 다른 자리에 위치하면 볼 한 번으로 센다.

예) 영수가 324를 갖고 있으면

  • 429는 1 스트라이크 1 볼이다.
  • 241은 0 스트라이크 2 볼이다.
  • 924는 2 스트라이크 0 볼이다.
  • 영수는 민혁이가 말한 수가 몇 스트라이크 몇 볼인지를 답해준다.
  • 민혁이가 영수의 세 자리 수를 정확하게 맞추어 3 스트라이크가 되면 게임이 끝난다. 아니라면 민혁이는 새로운 수를 생각해 다시 영수에게 묻는다.

현재 민혁이와 영수는 게임을 하고 있는 도중에 있다. 민혁이가 영수에게 어떤 수들을 물어보았는지, 그리고 각각의 물음에 영수가 어떤 대답을 했는지가 입력으로 주어진다. 이 입력을 바탕으로 여러분은 영수가 생각하고 있을 가능성이 있는 수가 총 몇 개인지를 알아맞혀야 한다.

아래와 같은 경우를 생각해보자.

  • 민혁: 123
  • 영수: 1 스트라이크 1 볼.
  • 민혁: 356
  • 영수: 1 스트라이크 0 볼.
  • 민혁: 327
  • 영수: 2 스트라이크 0 볼.
  • 민혁: 489
  • 영수: 0 스트라이크 1 볼.

이때 가능한 답은 324와 328, 이렇게 두 가지이다.

영수는 동아리의 규율을 잘 따르는 착한 아이라 민혁이의 물음에 곧이곧대로 정직하게 답한다. 그러므로 영수의 답들에는 모순이 없다.

민혁이의 물음들과 각각의 물음에 대한 영수의 답이 입력으로 주어질 때 영수가 생각하고 있을 가능성이 있는 답의 총 개수를 출력하는 프로그램을 작성하시오.

 

입력

첫째 줄에는 민혁이가 영수에게 몇 번이나 질문을 했는지를 나타내는 1 이상 100 이하의 자연수 N이 주어진다. 이어지는 N개의 줄에는 각 줄마다 민혁이가 질문한 세 자리 수와 영수가 답한 스트라이크 개수를 나타내는 정수와 볼의 개수를 나타내는 정수, 이렇게 총 세 개의 정수가 빈칸을 사이에 두고 주어진다.

 

출력

첫 줄에 영수가 생각하고 있을 가능성이 있는 답의 총 개수를 출력한다.

 

예제 입력

4
123 1 1
356 1 0
327 2 0
489 0 1

 

예제 출력

2

 

해결방법

모든 경우를 다 해보는 완전탐색 문제이다

 

먼저 시간 복잡도를 생각해봤다

  • 순서가 있는 경우를 순열 - 𝗻𝐏𝐫 : 서로 다른 n개 중 r개를 뽑아 나열하는 경우
  • 순서가 없는 경우를 조합 - 𝗻𝑪𝐫 : 서로 다른 n개 중 r개를 뽑는 경우

 

숫자 야구의 경우는 서로 다른 n개 중 r개를 뽑아 나열하는 순열의 경우이다

따라서 n! / (n-r)! 의 공식을 적용하면 50,400 이란 경우의 수가 나온다

이 때, 총 질문은 100번 할 수 있으니 504,000 이라는 시간으로 모든 경우를 탐색할 수 있다

 

 

먼저 3자리 순열을 만들어야 한다

이 작업은 재귀를 통해 구현하였다

// 3 자리 순열을 만들기 위한 DFS
void dfs(int cnt,string d){
    if(cnt == 3){
        if(solve(d)){
            ans++;
        }
        return;
    }
    
    for(int i=1; i<=9; i++){
        if(!visited[i]){
            visited[i] = true;
            dfs(cnt+1, d+to_string(i));
            visited[i] = false;
        }
    }
}

 

이제 3자리 순열이 완성되었다면 각각 질문에 대해 스트라이크, 볼을 계산한 뒤 비교해야한다

// 3자리 수와 질문의 수를 비교하여 스트라이크, 볼을 계산한 뒤 비교하는 함수
bool solve(string target){
    for(int i=0; i<question.size(); i++){
        string num = to_string(question[i].num);
        int strike = 0;
        int ball = 0;
        
        // 스트라이크 검사
        for(int j=0; j<3; j++){
            if(num[j] == target[j]) strike++;
        }
        
        // 볼 검사
        for(int j=0; j<3; j++){
            for(int k=0; k<3; k++){
                if(j==k) continue;
                
                if(num[k] == target[j]) ball++;
            }
        }
        
        // 스트라이크 또는 볼의 개수가 일치하지 않다면 false 리턴
        if(question[i].strike != strike || question[i].ball != ball)
            return false;
    }
    // 모든 질문에 대해 답이 일치하면 true 리턴
    return true;
}

 

 

 

소스코드

#include <iostream>
#include <stdio.h>
#include <math.h>
#include <cstring>
#include <set>
#include <queue>
#include <vector>
#include <algorithm>
#define MAX 0
#define INF 987654321
using namespace std;

struct number {
    int num,strike,ball;
    
    number() { }
    number(int n,int s,int b) : num(n),strike(s),ball(b) { }
};

int t,n,s,b,ans;
bool visited[11];

vector<number> question;

// 3자리 수와 질문의 수를 비교하여 스트라이크, 볼을 계산한 뒤 비교하는 함수
bool solve(string target){
    for(int i=0; i<question.size(); i++){
        string num = to_string(question[i].num);
        int strike = 0;
        int ball = 0;
        
        // 스트라이크 검사
        for(int j=0; j<3; j++){
            if(num[j] == target[j]) strike++;
        }
        
        // 볼 검사
        for(int j=0; j<3; j++){
            for(int k=0; k<3; k++){
                if(j==k) continue;
                
                if(num[k] == target[j]) ball++;
            }
        }
        
        // 스트라이크 또는 볼의 개수가 일치하지 않다면 false 리턴
        if(question[i].strike != strike || question[i].ball != ball)
            return false;
    }
    // 모든 질문에 대해 답이 일치하면 true 리턴
    return true;
}

// 3 자리 순열을 만들기 위한 DFS
void dfs(int cnt,string d){
    if(cnt == 3){
        if(solve(d)){
            ans++;
        }
        return;
    }
    
    for(int i=1; i<=9; i++){
        if(!visited[i]){
            visited[i] = true;
            dfs(cnt+1, d+to_string(i));
            visited[i] = false;
        }
    }
}

int main(int argc, const char * argv[]) {
    // cin,cout 속도향상
    ios_base::sync_with_stdio(false);
    cin.tie(NULL); cout.tie(NULL);
    
    cin >> t;
    for(int i=0; i<t; i++){
        cin >> n >> s >> b;
        question.push_back(number(n,s,b));
    }
    
    ans = 0;
    memset(visited, false, sizeof(visited));
    dfs(0,"");
    cout << ans << "\n";
    
    return 0;
}

 

'Algorithm > BOJ 문제풀이' 카테고리의 다른 글

[BFS] 2636번 치즈  (0) 2019.01.06
[Dijkstra] 2151번 거울 설치  (0) 2019.01.05
[BF] 1018번 체스판 다시 칠하기  (0) 2019.01.05
[BF] 10448번 유레카 이론  (0) 2019.01.04
[BF] 3085번 사탕 게임  (0) 2019.01.04