백준 1966 프린터 큐 [JAVA]

Ga0·2023년 7월 1일
1

baekjoon

목록 보기
77/139

문제 해석

  • 문제를 해석하면, 가장 첫째 줄에는 테스트케이스의 수(T)를 입력받는다.
  • 테스트케이스의 수(T)를 입력받았다면, 각각의 테스트케이스를 입력받는데, 테스트케이스 한 개마다 2줄을 입력받는다.
  • 일단, 테스트 케이스의 첫째 줄에는 문서의 개수(N)과 몇번째로 인쇄되는지 궁금한 문서가 몇번째 순서(M)인지 입력받는다.
  • 첫째 줄을 모두 입력 받고 나면, 각각의 문서의 중요도인가 차례대로 주어진다.
  • 문서의 중요도는 1이상이고 9 이하이다. (즉, 중요도가 같은 문서는 여러 개 존재할 수 있다.)
  • 문서의 중요도가 같은 경우는 맨 앞에 있는 것을 먼저 뺀다. (여기서 주의 해야할 점이 있는데, 중요도가 현재 맨앞에 있는 요소보다 큰 것이 있으면 맨 뒤로 보낸다는 것! 즉, 순서는 바뀔 수 있다는 것이다.)
  • 말로만 설명하면, 어려운데 그림을 설명하면 아래와 같다.

코드

import java.io.*;
import java.util.*;

public class Main {

    static LinkedList<int[]> queue; // 문서들의 중요도를 저장하는 큐(초기 인덱스 값, 중요도)

    public static void main(String[] args) throws IOException {
        BufferedReader br = new BufferedReader(new InputStreamReader(System.in));
        StringBuilder sb = new StringBuilder();
        StringTokenizer st;

        int T = Integer.parseInt(br.readLine()); // 테스트케이스의 개수

        while(T --> 0){ //테스트케이스의 개수만큼 반복
            queue = new LinkedList<>(); //테스트케이스만큼 큐가 있는거나 다름없으니 반복문 돌때마다 초기화해준다.

            st = new StringTokenizer(br.readLine());

            int N = Integer.parseInt(st.nextToken()); //문서의 개수
            int M = Integer.parseInt(st.nextToken()); //몇번째로 인쇄되는지 알고 싶은 문서의 순서 번호

            st = new StringTokenizer(br.readLine());

            for(int i = 0; i < N; i++){ //문서의 개수만큼 입력받아야 하므로 N번 반복
                queue.add(new int[]{i, Integer.parseInt(st.nextToken())}); //각각의 초기 위치, 중요도를 입력한다.
            }

            sb.append(solution(M)).append("\n");
        }
        br.close();
        
        System.out.println(sb);
    }

    // 해당 M번째 문서가 몇번째로 인쇄되는지 구하는 메소드(매개변수 M)
    static int solution(int M){
        int findIt = 0; //M이 몇번째로 인쇄되었는지 저장하는 변수

        while(!queue.isEmpty()){ //큐가 비어있을 때까지 반복한다.

            int[] first = queue.poll(); //큐의 맨 앞 요소를 뺀다.
            boolean isMax = true; //해당 요소가 가장 큰지 유효성 검사하는 변수

            //큐에 남아있는 모든 요소들과 반복문을 돌력중요도를 비교한다.
            for(int i = 0; i < queue.size(); i++){

                //지금 뽑은 요소보다 큰 요소다 있을 경우
                if(first[1] < queue.get(i)[1]){

                    queue.offer(first); //뽑은 해당 요소를 맨 뒤에 넣고

                    //반복하면서 마주한 뽑은 요소(first)보다 안 큰 요소들 또한 맨뒤에 넣는다.
                    for(int j = 0; j < i; j++){
                        //queue의 i번째에 first(뽑은 요소)보다 큰 게 있는 것이므로
                        // i까지만 반복한다.
                        queue.offer(queue.poll());
                    }

                    isMax = false; // 해당 뽑은 요소(first)보다 큰 요소가 있다는 뜻이므로 first를 한다.
                    break; // 걸러졌으므로 반복문을 나온다.
                }
            }

            if(isMax == false){ //뽑은 요소보다 큰 요소가 있어서 걸러졌다면 다시 반복문을 돌아야한다.(밑은 수행X)
                continue;
            }

            //first가 가장 큰 요소인 거니까
            findIt++; //1씩 증가한다.(큰 요소를 하나씩 누적 개수 더하기 해서)

            if(first[0] == M){ //찾는 요소일 경우 반복문을 빠져나가서
                break;
            }
        }

        return findIt; //큰 요소를 누적 개서 더하기 한거를 반환
    }
}
  • 개인적으로 너무 어려웠다.
  • 문제는 이해했지만, 코드로 전환하는 게 생각이 나질 않아서 참고 블로그 참고 했다...
  • 그래서 이 문제는 이해는 했지만 직접 코드를 생각해서 치지 못한 문제이다...
  • 아무튼, 코드 설명은 주석으로 작성해두었다.

결과

느낀 점

  • 문제 해석에 시간을 코딩할 때 보다 더 쓴 것 같다. (내 문해력이 부족하다는 것을 느꼈다.)
  • 코드를 이해했지만 처음부터 직접 작성하지 못해서 다시 리뷰해야할 문제...(하...)

0개의 댓글