문제

N×N 크기의 공간에 물고기 M마리와 아기 상어 1마리가 있다. 공간은 1×1 크기의 정사각형 칸으로 나누어져 있다. 한 칸에는 물고기가 최대 1마리 존재한다.

아기 상어와 물고기는 모두 크기를 가지고 있고, 이 크기는 자연수이다. 가장 처음에 아기 상어의 크기는 2이고, 아기 상어는 1초에 상하좌우로 인접한 한 칸씩 이동한다.

아기 상어는 자신의 크기보다 큰 물고기가 있는 칸은 지나갈 수 없고, 나머지 칸은 모두 지나갈 수 있다. 아기 상어는 자신의 크기보다 작은 물고기만 먹을 수 있다. 따라서, 크기가 같은 물고기는 먹을 수 없지만, 그 물고기가 있는 칸은 지나갈 수 있다.

아기 상어가 어디로 이동할지 결정하는 방법은 아래와 같다.

  • 더 이상 먹을 수 있는 물고기가 공간에 없다면 아기 상어는 엄마 상어에게 도움을 요청한다.
  • 먹을 수 있는 물고기가 1마리라면, 그 물고기를 먹으러 간다.
  • 먹을 수 있는 물고기가 1마리보다 많다면, 거리가 가장 가까운 물고기를 먹으러 간다.
    • 거리는 아기 상어가 있는 칸에서 물고기가 있는 칸으로 이동할 때, 지나야하는 칸의 개수의 최솟값이다.
    • 거리가 가까운 물고기가 많다면, 가장 위에 있는 물고기, 그러한 물고기가 여러마리라면, 가장 왼쪽에 있는 물고기를 먹는다.

아기 상어의 이동은 1초 걸리고, 물고기를 먹는데 걸리는 시간은 없다고 가정한다. 즉, 아기 상어가 먹을 수 있는 물고기가 있는 칸으로 이동했다면, 이동과 동시에 물고기를 먹는다. 물고기를 먹으면, 그 칸은 빈 칸이 된다.

아기 상어는 자신의 크기와 같은 수의 물고기를 먹을 때 마다 크기가 1 증가한다. 예를 들어, 크기가 2인 아기 상어는 물고기를 2마리 먹으면 크기가 3이 된다.

공간의 상태가 주어졌을 때, 아기 상어가 몇 초 동안 엄마 상어에게 도움을 요청하지 않고 물고기를 잡아먹을 수 있는지 구하는 프로그램을 작성하시오.

 

입력

첫째 줄에 공간의 크기 N(2 ≤ N ≤ 20)이 주어진다.

둘째 줄부터 N개의 줄에 공간의 상태가 주어진다. 공간의 상태는 0, 1, 2, 3, 4, 5, 6, 9로 이루어져 있고, 아래와 같은 의미를 가진다.

  • 0: 빈 칸
  • 1, 2, 3, 4, 5, 6: 칸에 있는 물고기의 크기
  • 9: 아기 상어의 위치

아기 상어는 공간에 한 마리 있다.

 

출력

첫째 줄에 아기 상어가 엄마 상어에게 도움을 요청하지 않고 물고기를 잡아먹을 수 있는 시간을 출력한다.

 

예제

풀이 방법

이 문제는 저번에 도전했으나 해결하지 못해서 다시 풀어본 문제입니다.

 

이 문제에서 아기 상어가 어디로 이동해야 할지 결정하는 방법은

1. 가장 가까운 물고기를 잡아먹되

2. 가까운 물고기가 1마리 이상이라면 가장 위의 있는 물고기

3. 2번 조건을 만족하는 물고기가 1마리 이상이라면 가장 왼쪽의 있는 물고기를 먹기

위의 조건을 만족하는 물고기를 잡아먹어야 합니다.

 

여기서 주의해야 할 점은 아기 상어보다 크기가 큰 물고기는 지나갈 수 없고, 크기가 같은 물고기는 지나갈 수는 있으나 잡아먹을 순 없다. 즉, 아기 상어보다 크기가 작은 물고기만 잡아먹을 수 있다는 점.

또한 아기 상어의 크기만큼 물고기를 잡아먹었다면 아기 상어 크기가 1 증가한다는 점입니다.

 

제가 푼 방법은

1. 아기 상어 위치 shark 큐에 넣기

2. 아기 상어를 상 하 좌 우로 움직이면서 먹을 수 있는 먹이 정보 feed 큐에 넣기

3. feed 큐에 들어 있는 먹이들 중 거리가 가장 가까운 먹이 먹기 = (먹이 위치 0으로 변경)

4. 해당 먹이 위치로 아기 상어 이동, 잡아먹은 횟수 1 증가, 먹이 큐 초기화, 아기 상어가 먹은 물고기 수 확인하여 아기 상어 크기 1 증가

해당 먹이가 없을 때까지 1 ~ 4번 반복해주었습니다.

 

 

코드

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
import java.util.Comparator;
import java.util.LinkedList;
import java.util.PriorityQueue;
import java.util.Queue;
import java.util.StringTokenizer;
 
public class Main_BJ_16236_아기상어 {
    
    static class info{
        int x,y,distance;
 
        public info(int x, int y, int distance) {
            super();
            this.x = x;
            this.y = y;
            this.distance = distance;
        }
        
    }
    
    static int [] dx = {-1,1,0,0}; //상 하 좌 우
    static int [] dy = {0,0,-1,1};
    
    static int N, shark_age, size=0, count=0, time=0;
    static int [][] map;
    static PriorityQueue<info> feed = new PriorityQueue<>(new Comparator<info>() {
        //가장 가깝고, 가장 위, 가장 왼쪽
        @Override
        public int compare(info o1, info o2) {
            if(o1.distance==o2.distance) {
                if(o1.x==o2.x) return o1.y-o2.y;
                else return o1.x-o2.x;
            }
            return o1.distance-o2.distance;
        }
    });
    
    static Queue<info> shark = new LinkedList<>();
    
    public static void main(String[] args) throws NumberFormatException, IOException {
        BufferedReader br = new BufferedReader(new InputStreamReader(System.in));
        StringTokenizer st;
        
        N = Integer.parseInt(br.readLine());
        
        map = new int[N][N];
        
        for(int i=0;i<N;i++) {
            st = new StringTokenizer(br.readLine());
            for(int j=0;j<N;j++) {
                map[i][j] = Integer.parseInt(st.nextToken());
                if(map[i][j]==9) {
                    shark.add(new info(i,j,0));
                    map[i][j] = 0;
                }
            }
        }
        shark_age=2;
        findFeed();
 
    }
    
    public static void findFeed() {
        
        while(true) {
            
            boolean [][] visited = new boolean[N][N];
            
            while(!shark.isEmpty()) {
                info temp = shark.poll();
                visited[temp.x][temp.y] = true;
                
                for(int i=0;i<4;i++) {
                    int nx = temp.x+dx[i];
                    int ny = temp.y+dy[i];
                    
                    if(range(nx,ny) && !visited[nx][ny]) {
                        if(0<map[nx][ny] && map[nx][ny]<7) {
                            if(map[nx][ny]<shark_age) {
                                feed.add(new info(nx,ny,temp.distance+1));
                                shark.add(new info(nx,ny,temp.distance+1));
                                visited[nx][ny]=true;
                            }
                            else if(map[nx][ny]==shark_age) {
                                shark.add(new info(nx,ny,temp.distance+1));
                                visited[nx][ny]=true;
                            }
                        }
                        else if(map[nx][ny]==0) {
                            shark.add(new info(nx,ny,temp.distance+1));
                            visited[nx][ny]=true;
                        }
                    }
                }
            }
            
            //가장 가까운 물고기 먹고
            if(!feed.isEmpty()) eat();
            else break;
        }
        System.out.println(time);
        
    }
    
    public static boolean range(int x, int y) {
        return x>=0 && x<&& y>=0 && y<N;
    }
    
    public static void eat() {
        info eat = feed.poll();
        
        count++;
        if(count==shark_age) {
            shark_age++;
            count=0;
        }
        
        shark.add(new info(eat.x,eat.y,0));
        time+=eat.distance;
        map[eat.x][eat.y]=0;
        feed.clear();
    }
 
}
 
cs

 

'문제 > 백준' 카테고리의 다른 글

백준 1520 내리막길 [JAVA]  (0) 2021.09.10
백준 17121 연구소2 [JAVA]  (0) 2021.09.02
백준 9081번 단어 맞추기 [JAVA]  (0) 2021.07.08
백준 1713번 후보 추천하기 [JAVA]  (1) 2021.07.04
백준 4358번 생태학 [JAVA]  (0) 2021.06.30

문제

BEER라는 단어를 이루는 알파벳들로 만들 수 있는 단어들을 사전 순으로 정렬하게 되면

와 같이 된다. 이러한 순서에서 BEER 다음에 오는 단어는 BERE가 된다. 이와 같이 단어를 주면 그 단어를 이루는 알파벳들로 만들 수 있는 단어들을 사전 순으로 정렬할 때에 주어진 단어 다음에 나오는 단어를 찾는 프로그램을 작성하시오.

 

입력

입력의 첫 줄에는 테스트 케이스의 개수 T (1 ≤ T ≤ 10)가 주어진다. 각 테스트 케이스는 하나의 단어가 한 줄로 주어진다. 단어는 알파벳 A~Z 대문자로만 이루어지며 항상 공백이 없는 연속된 알파벳으로 이루어진다. 단어의 길이는 100을 넘지 않는다.

 

출력

각 테스트 케이스에 대해서 주어진 단어 바로 다음에 나타나는 단어를 한 줄에 하나씩 출력하시오. 만일 주어진 단어가 마지막 단어이라면 그냥 주어진 단어를 출력한다.

 

예제

풀이 방법

주어진 단어 다음에 나오는 단어를 찾는 문제였습니다.

 

처음에는 가능한 단어들을 다 만들어 놓고 정렬한 뒤, 주어진 단어의 다음 단어를 출력해주도록 구현했지만, 단어의 길이가 최대 99이므로 시간초과가 발생했습니다.

 

그래서 두번째로 구현한 방법이 

1 ) 단어 맨뒤에서부터 확인하면서 감소하는 부분 찾기

2 ) 다시 단어 맨뒤에서부터 확인하면서 감소하는 부분보다 큰 부분 찾기

3 ) 둘 위치 바꿔주기

4 ) 1)의 위치 뒷 부분 정렬

입니다.

 

예시)

 

코드

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
import java.util.Arrays;
 
public class Main_BJ_9081_단어맞추기 {
    
    public static void main(String[] args) throws IOException {
        BufferedReader br = new BufferedReader(new InputStreamReader(System.in));
        StringBuilder sb = new StringBuilder();
        
        int T = Integer.parseInt(br.readLine());
        
        for(int i=0;i<T;i++) {
            char [] word = br.readLine().toCharArray();
            
            int index=-1, index2=0;
            char temp;
        
            for(int j=word.length-1;j>0;j--) {
                if(word[j-1]<word[j]) {
                    index=j-1break;
                }
            }
            
            if(index==-1) {
                for(int j=0;j<word.length;j++) sb.append(word[j]);
                sb.append("\n");
            }
            else {
                for(int j=word.length-1;j>=0;j--) {
                    if(word[index]<word[j]) {
                        index2=j; break;
                    }
                }
                temp = word[index];
                word[index] = word[index2];
                word[index2] = temp;
                
                Arrays.sort(word,index+1, word.length);
                
                for(int j=0;j<word.length;j++) sb.append(word[j]);
                sb.append("\n");
            }
            
        }
        System.out.println(sb.toString());
    }
}
cs

 

 

문제

월드초등학교 학생회장 후보는 일정 기간 동안 전체 학생의 추천에 의하여 정해진 수만큼 선정된다. 그래서 학교 홈페이지에 추천받은 학생의 사진을 게시할 수 있는 사진틀을 후보의 수만큼 만들었다. 추천받은 학생의 사진을 사진틀에 게시하고 추천받은 횟수를 표시하는 규칙은 다음과 같다.

  1. 학생들이 추천을 시작하기 전에 모든 사진틀은 비어있다.
  2. 어떤 학생이 특정 학생을 추천하면, 추천받은 학생의 사진이 반드시 사진틀에 게시되어야 한다.
  3. 비어있는 사진틀이 없는 경우에는 현재까지 추천 받은 횟수가 가장 적은 학생의 사진을 삭제하고, 그 자리에 새롭게 추천받은 학생의 사진을 게시한다. 이때, 현재까지 추천 받은 횟수가 가장 적은 학생이 두 명 이상일 경우에는 그러한 학생들 중 게시된 지 가장 오래된 사진을 삭제한다.
  4. 현재 사진이 게시된 학생이 다른 학생의 추천을 받은 경우에는 추천받은 횟수만 증가시킨다.
  5. 사진틀에 게시된 사진이 삭제되는 경우에는 해당 학생이 추천받은 횟수는 0으로 바뀐다.

후보의 수 즉, 사진틀의 개수와 전체 학생의 추천 결과가 추천받은 순서대로 주어졌을 때, 최종 후보가 누구인지 결정하는 프로그램을 작성하시오.

 

입력

첫째 줄에는 사진틀의 개수 N이 주어진다. (1 ≤ N ≤ 20) 둘째 줄에는 전체 학생의 총 추천 횟수가 주어지고, 셋째 줄에는 추천받은 학생을 나타내는 번호가 빈 칸을 사이에 두고 추천받은 순서대로 주어진다. 총 추천 횟수는 1,000번 이하이며 학생을 나타내는 번호는 1부터 100까지의 자연수이다.

 

출력

사진틀에 사진이 게재된 최종 후보의 학생 번호를 증가하는 순서대로 출력한다.

 

예제

풀이 방법

전체 학생의 추천 결과가 추천받은 순서대로 주어졌을 때, 최종 후보의 학생 번호를 출력하는 문제였습니다.

 

먼저, 후보의 사진을 게시할 수 있는 규칙은

1) N개의 사진틀 중, 비어있는 사진 틀이 존재한다면 학생 사진 게시

2) 비어있는 사진 틀이 존재하지 않는다면

   i) 추천 받은 횟수가 가장 적은 학생 사진 삭제 후 추천 학생 사진 게시

   ii) 추천 받은 횟수가 같으면 게시된 지 가장 오래된 사진 삭제 후 추천 학생 사진 게시

단) 현재 사진이 게시된 학생이 추천을 받은 경우에는 횟수 증가

로 정리할 수 있습니다.

 

게시할 사진의 정보를 저장할 list를 이용하여 만약 list의 사이즈가 N개 보다 적을 경우에는 list안에 추천받은 학생의 정보가 있는지 확인 후, 만약 존재한다면 횟수를 +1 증가시키고, 정보가 없는 경우에는 list에 학생정보를 넣어주었습니다. 만약 비어있는 사진틀이 존재하지 않는다면 마찬가지로 list안에 추천받은 학생의 정보가 있는 경우에는 추천 횟수를 +1 증가시키고, 없는 경우에는 list 맨 앞에 있는 정보를 삭제 후, 추천받은 학생의 정보를 넣어주었습니다.

마지막으로 최종 후보의 학생 번호를 출력하기 위해, list에 들어가있는 학생의 번호를 answer 리스트에 넣어준 뒤, 학생 번호순으로 출력해주었습니다.

 

** 처음에 생각하지 못했던 부분이 비어있는 사진틀이 존재할 경우에 현재 사진이 게시된 학생인지 확인하지 않고 리스트에 넣어줘서 자꾸 틀렸습니다가 떴다.. 앞으로 생각좀하면서 문제풀어야지..

 

코드

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
import java.util.ArrayList;
import java.util.Collections;
import java.util.StringTokenizer;
 
public class Main_BJ_1713_후보추천하기 {
    
    static class info implements Comparable<info>{
        int index, num, cnt;
 
        public info(int index, int num, int cnt) {
            super();
            this.index = index;
            this.num = num;
            this.cnt = cnt;
        }
 
        @Override
        public int compareTo(info o) {
            if(this.cnt==o.cnt) {
                return this.index-o.index;
            }
            return this.cnt-o.cnt;
        }
        
    }
 
    public static void main(String[] args) throws NumberFormatException, IOException {
        BufferedReader br = new BufferedReader(new InputStreamReader(System.in));
        StringTokenizer st;
        StringBuilder sb = new StringBuilder();
        
        ArrayList<info> list = new ArrayList<>();
        ArrayList<Integer> answer = new ArrayList<>();
        
        int N = Integer.parseInt(br.readLine()); //사진 틀
        int M = Integer.parseInt(br.readLine()); //총 추천 횟수
        
        st = new StringTokenizer(br.readLine());
        for(int i=0;i<M;i++) {
            int student = Integer.parseInt(st.nextToken());
            if(list.size()<N) {
                boolean flag = false;
                for(int j=0;j<list.size();j++) {
                    if(list.get(j).num==student) {
                        list.get(j).cnt++;
                        flag = truebreak;
                    }
                }
                if(!flag) list.add(new info(i,student,1));
            }
            else {
                Collections.sort(list);
                boolean flag = false;
                for(int j=0;j<list.size();j++) {
                    if(list.get(j).num==student) {
                        list.get(j).cnt++;
                        flag = true;
                        break;
                    }
                }
                if(!flag) {
                    list.remove(0);
                    list.add(new info(i,student,1));
                }
            }
        }
        
        for(int i=0;i<list.size();i++) {
            answer.add(list.get(i).num);
        }
        
        Collections.sort(answer);
        
        for(int i=0;i<answer.size();i++) sb.append(answer.get(i)+" ");
        
        System.out.println(sb.toString());    
    }
 
}
cs

 

문제

생태학에서 나무의 분포도를 측정하는 것은 중요하다. 그러므로 당신은 미국 전역의 나무들이 주어졌을 때, 각 종이 전체에서 몇 %를 차지하는지 구하는 프로그램을 만들어야 한다.

 

입력

프로그램은 여러 줄로 이루어져 있으며, 한 줄에 하나의 나무 종 이름이 주어진다. 어떤 종 이름도 30글자를 넘지 않으며, 입력에는 최대 10,000개의 종이 주어지고 최대 1,000,000그루의 나무가 주어진다.

 

출력

주어진 각 종의 이름을 사전순으로 출력하고, 그 종이 차지하는 비율을 백분율로 소수점 4째자리까지 반올림해 함께 출력한다.

 

예제

 

풀이 방법

나무들이 주어졌을 때, 각 나무의 종이 전체에서 몇 %를 차지하는지 구하는 문제였습니다.

이 문제를 읽어봤을 때, HashMap을 이용해서 풀면 되겠구나 라는 생각을 먼저 했습니다. 나무의 종을 입력받고 만약 HashMap의 key 값에 tree가 포함되어 있지 않을 경우에는 같은 종이 없다는 의미므로 <tree 명, 1> 쌍의 <키, 값>을 HashMap에 넣어주었습니다. 반대로 이미 있는 경우에는 value값만 +1 해주었습니다. 나무를 다 입력받고 난 후, 사전 순으로 정렬하기 위해 ArrayList에 HashMap에 넣었던 데이터들을 넣어주고 정렬해주었습니다. 마지막으로 전체에서 얼마나 차지하는지를 구하기 위해 계산해주고 출력해주었습니다.

 

이 문제를 풀면서 NullPointer 런타임 에러가 발생했습니다.ㅠㅠㅠ 에러를 찾던 중, 32번 줄에서 발생한다는 것을 알고 수정했더니 통과할 수 있었습니다.ㅠㅠㅠㅠ

 

코드

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
import java.util.ArrayList;
import java.util.Collections;
import java.util.Comparator;
import java.util.HashMap;
import java.util.Iterator;
 
public class Main_BJ_4358_생태학 {
    
    static class info{
        String tree;
        int cnt;
        
        public info(String tree, int cnt) {
            this.tree = tree;
            this.cnt = cnt;
        }    
    }
 
    public static void main(String[] args) throws IOException {
        BufferedReader br = new BufferedReader(new InputStreamReader(System.in));
        StringBuilder sb = new StringBuilder();
        
        HashMap<String, Integer> hm = new HashMap<String, Integer>();
        
        int count=0;
        while(true) {
            String tree = br.readLine();
            
            if(tree==null || tree.equals("")) break;
            count++;
            if(!hm.containsKey(tree)) hm.put(tree,1);
            else hm.put(tree, hm.get(tree)+1);
        }
        
        ArrayList<info> list = new ArrayList<>();
        Iterator<String> keys = hm.keySet().iterator();
        while(keys.hasNext()) {
            String tree1 = keys.next();
            list.add(new info(tree1,hm.get(tree1)));
        }
        
        Collections.sort(list, new Comparator<info>() {
 
            @Override
            public int compare(info o1, info o2) {
                return o1.tree.compareTo(o2.tree);
            }
        });
        
        for(int i=0;i<list.size();i++) {
            double per = (double)(list.get(i).cnt*100.0)/count;
            
            sb.append(list.get(i).tree+" "+String.format("%.4f", per)+"\n");
        }
        System.out.println(sb.toString());
    }
 
}
cs

 

문제

한윤정과 친구들은 이탈리아로 방학 여행을 갔다. 이탈리아는 덥다. 윤정이와 친구들은 아이스크림을 사먹기로 했다. 아이스크림 가게에는 N종류의 아이스크림이 있다. 모든 아이스크림은 1부터 N까지 번호가 매겨져있다. 어떤 종류의 아이스크림을 함께먹으면, 맛이 아주 형편없어진다. 따라서 윤정이는 이러한 경우를 피하면서 아이스크림을 3가지 선택하려고 한다. 이때, 선택하는 방법이 몇 가지인지 구하려고 한다.

 

입력

첫째 줄에 정수 N과 M이 주어진다. N은 아이스크림 종류의 수이고, M은 섞어먹으면 안 되는 조합의 개수이다. 아래 M개의 줄에는 섞어먹으면 안 되는 조합의 번호가 주어진다. 같은 조합은 두 번 이상 나오지 않는다. (1 ≤ N ≤ 200, 0 ≤ M ≤ 10,000)

 

출력

첫째 줄에, 가능한 방법이 총 몇 개 있는지 출력한다.

 

예제

 

풀이 방법

이 문제는 섞어먹으면 안 되는 조합을 피하면서 아이스크림을 3가지 선택하는 방법의 수를 구하는 문제였습니다.

먼저 피해야하는 조합의 번호인 값을 icecream 2차원 배열에 true로 표시해주었습니다. 그리고 난 후, 아이스크림 3가지를 선택하기 위해 조합을 이용한 뒤, 3가지를 모두 선택한 후, 이 아이스크림이 섞어먹어도 괜찮은 조합인지 판단하기 위해 isCheck() 메서드를 통해 확인해 주었습니다. 만약 고른 아이스크림 조합의 icecream 배열 값이 true인 경우에는 섞어먹으면 안 되는 조합이 포함되어있는 경우이므로 false를 리턴해주었습니다. 반대로 for문을 다 끝냈다면 이 아이스크림 조합은 섞어먹어도 괜찮은 조합이므로 true를 리턴해 선택하는 방법의 수인 ans값을 +1 해주었습니다. 모든 조합의 결과를 확인 한 뒤, 총 선택할 수 있는 방법의 수를 출력해주었습니다.

 

코드

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
import java.util.StringTokenizer;
 
public class Main_BJ_2242_한윤정이이탈리아에가서아이스크림을사먹는데 {
    
    static int N,M,ans=0;
    static boolean [][] icecream;
    static boolean [] visited;
    static int [] select;
 
    public static void main(String[] args) throws IOException {
        BufferedReader br = new BufferedReader(new InputStreamReader(System.in));
        StringTokenizer st;
        
        st = new StringTokenizer(br.readLine());
        N = Integer.parseInt(st.nextToken());
        M = Integer.parseInt(st.nextToken());
        
        icecream = new boolean[N][N];
        visited = new boolean[N];
        select = new int[3];
        
        for(int i=0;i<M;i++) {
            st = new StringTokenizer(br.readLine());
            int a = Integer.parseInt(st.nextToken())-1;
            int b = Integer.parseInt(st.nextToken())-1;
            
            icecream[a][b] = icecream[b][a] = true;
        }
        
        comb(0,0);
        System.out.println(ans);
 
    }
    
    public static void comb(int start,int cnt) {
        if(cnt==3) {
            if(isCheck()) ans++;
            return;
        }
        
        for(int i=start;i<N;i++) {
            if(!visited[i]) {
                visited[i] = true;
                select[cnt] = i;
                comb(i,cnt+1);
                visited[i] = false;
            }
        }
        
    }
    
    public static boolean isCheck() {
        for(int i=0;i<3;i++) {
            for(int j=i+1;j<3;j++) {
                if(icecream[select[i]][select[j]]) return false;
            }
        }
        return true;
    }
}
cs

 

 

'문제 > 백준' 카테고리의 다른 글

백준 1713번 후보 추천하기 [JAVA]  (1) 2021.07.04
백준 4358번 생태학 [JAVA]  (0) 2021.06.30
백준 1043번 거짓말 [JAVA]  (0) 2021.06.25
백준 2174번 로봇 시뮬레이션 [JAVA]  (0) 2021.06.25
백준 2252번 줄세우기 [JAVA]  (0) 2021.06.18

문제

지민이는 파티에 가서 이야기 하는 것을 좋아한다. 파티에 갈 때마다, 지민이는 지민이가 가장 좋아하는 이야기를 한다. 지민이는 그 이야기를 말할 때, 있는 그대로 진실로 말하거나 엄청나게 과장해서 말한다. 당연히 과장해서 이야기하는 것이 훨씬 더 재미있기 때문에, 되도록이면 과장해서 이야기하려고 한다. 하지만, 지민이는 거짓말쟁이로 알려지기는 싫어한다. 문제는 몇몇 사람들은 그 이야기의 진실을 안다는 것이다. 따라서 이런 사람들이 파티에 왔을 때는, 지민이는 진실을 이야기할 수 밖에 없다. 당연히, 어떤 사람이 어떤 파티에서는 진실을 듣고, 또다른 파티에서는 과장된 이야기를 들었을 때도 지민이는 거짓말쟁이로 알려지게 된다. 지민이는 이런 일을 모두 피해야 한다.

사람의 수 N이 주어진다. 그리고 그 이야기의 진실을 아는 사람이 주어진다. 그리고 각 파티에 오는 사람들의 번호가 주어진다. 지민이는 모든 파티에 참가해야 한다. 이때, 지민이가 거짓말쟁이로 알려지지 않으면서, 과장된 이야기를 할 수 있는 파티 개수의 최댓값을 구하는 프로그램을 작성하시오.

 

입력

첫째 줄에 사람의 수 N과 파티의 수 M이 주어진다.

둘째 줄에는 이야기의 진실을 아는 사람의 수와 번호가 주어진다. 진실을 아는 사람의 수가 먼저 주어지고 그 개수만큼 사람들의 번호가 주어진다. 사람들의 번호는 1부터 N까지의 수로 주어진다.

셋째 줄부터 M개의 줄에는 각 파티마다 오는 사람의 수와 번호가 같은 방식으로 주어진다.

N, M은 50 이하의 자연수이고, 진실을 아는 사람의 수와 각 파티마다 오는 사람의 수는 모두 0 이상 50 이하의 정수이다.

 

출력

첫째 줄에 문제의 정답을 출력한다.

 

예제

코드

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
import java.util.ArrayList;
import java.util.LinkedList;
import java.util.Queue;
import java.util.StringTokenizer;
 
public class Main_BJ_1043_거짓말 {
    
    static int N,M;
    static int [] truth;
    
    static ArrayList<Integer> party[], people[];
 
    public static void main(String[] args) throws IOException {
        BufferedReader br = new BufferedReader(new InputStreamReader(System.in));
        StringTokenizer st;
        
        st = new StringTokenizer(br.readLine());
        N = Integer.parseInt(st.nextToken()); //사람 수
        M = Integer.parseInt(st.nextToken()); // 파티의 수
        
        truth = new int[N+1];
        
        people = new ArrayList[N+1];
        party = new ArrayList[M];
        
        for(int i=1;i<N+1;i++) people[i] = new ArrayList<>();
        for(int i=0;i<M;i++) party[i] = new ArrayList<>();
        
        st = new StringTokenizer(br.readLine());
        
        int know = Integer.parseInt(st.nextToken());
        for(int i=0;i<know;i++) {
            int num = Integer.parseInt(st.nextToken());
            truth[num]= -1;
        }
        
        for(int i=0;i<M;i++) {
            st = new StringTokenizer(br.readLine());
            int person = Integer.parseInt(st.nextToken());
            for(int j=0;j<person;j++) {
                int num = Integer.parseInt(st.nextToken());
                party[i].add(num);
                people[num].add(i);
            }
        }
        
        isLie();
        
    }
 
    public static void isLie() {
        Queue<Integer> queue = new LinkedList<Integer>();
        boolean [] visited= new boolean[M];
        
        for(int i=1;i<N+1;i++) {
            if(truth[i]==-1) {
                for(int j=0;j<people[i].size();j++) {
                    if(!visited[people[i].get(j)]) {
                        queue.add(people[i].get(j));
                        visited[people[i].get(j)]=true;
                    }
                }
            }
        }
        
        while(!queue.isEmpty()) {
            int p = queue.poll();
            
             for(int i=0;i<party[p].size();i++) {
                 for(int j=0;j<people[party[p].get(i)].size();j++) {
                     if(!visited[people[party[p].get(i)].get(j)]) {
                         visited[people[party[p].get(i)].get(j)] = true;
                         queue.add(people[party[p].get(i)].get(j));
                     }
                 }
             }
        }
        int ans=0;
        for(int i=0;i<M;i++) {
            if(!visited[i]) {
                ans++;
            }
        }
        System.out.println(ans);
    }
}
cs

 

문제

가로 A(1≤A≤100), 세로 B(1≤B≤100) 크기의 땅이 있다. 이 땅 위에 로봇들이 N(1≤N≤100)개 있다.

로봇들의 초기 위치는 x좌표와 y좌표로 나타난다. 위의 그림에서 보듯 x좌표는 왼쪽부터, y좌표는 아래쪽부터 순서가 매겨진다. 또한 각 로봇은 맨 처음에 NWES 중 하나의 방향을 향해 서 있다. 초기에 서 있는 로봇들의 위치는 서로 다르다.

이러한 로봇들에 M(1≤M≤100)개의 명령을 내리려고 한다. 각각의 명령은 순차적으로 실행된다. 즉, 하나의 명령을 한 로봇에서 내렸으면, 그 명령이 완수될 때까지 그 로봇과 다른 모든 로봇에게 다른 명령을 내릴 수 없다. 각각의 로봇에 대해 수행하는 명령은 다음의 세 가지가 있다.

  1. L: 로봇이 향하고 있는 방향을 기준으로 왼쪽으로 90도 회전한다.
  2. R: 로봇이 향하고 있는 방향을 기준으로 오른쪽으로 90도 회전한다.
  3. F: 로봇이 향하고 있는 방향을 기준으로 앞으로 한 칸 움직인다.

간혹 로봇들에게 내리는 명령이 잘못될 수도 있기 때문에, 당신은 로봇들에게 명령을 내리기 전에 한 번 시뮬레이션을 해 보면서 안전성을 검증하려 한다. 이를 도와주는 프로그램을 작성하시오.

잘못된 명령에는 다음의 두 가지가 있을 수 있다.

  1. Robot X crashes into the wall: X번 로봇이 벽에 충돌하는 경우이다. 즉, 주어진 땅의 밖으로 벗어나는 경우가 된다.
  2. Robot X crashes into robot Y: X번 로봇이 움직이다가 Y번 로봇에 충돌하는 경우이다.

 

입력

첫째 줄에 두 정수 A, B가 주어진다. 다음 줄에는 두 정수 N, M이 주어진다. 다음 N개의 줄에는 각 로봇의 초기 위치(x, y좌표 순) 및 방향이 주어진다. 다음 M개의 줄에는 각 명령이 명령을 내리는 순서대로 주어진다. 각각의 명령은 명령을 내리는 로봇, 명령의 종류(위에 나와 있는), 명령의 반복 회수로 나타낸다. 각 명령의 반복 회수는 1이상 100이하이다.

 

 

출력

첫째 줄에 시뮬레이션 결과를 출력한다. 문제가 없는 경우에는 OK를, 그 외의 경우에는 위의 형식대로 출력을 한다. 만약 충돌이 여러 번 발생하는 경우에는 가장 먼저 발생하는 충돌을 출력하면 된다.

 

 

예제

 

풀이 방법

이 문제를 로봇들의 안정성을 검증하는 문제였습니다.

 

로봇이 수행할 수 있는 명령은 총 3가지로,

1) L : 로봇이 향하고 있는 방향을 기준으로 왼쪽으로 90도 회전

2) R : 로봇이 향하고 있는 방향을 기준으로 오른쪽으로 90도 회전

3) F : 로봇이 향하고 있는 방향을 기준으로 한 칸 이동 

 

먼저 로봇의 번호와 위치정보, 방향 정보를 list에 넣어두었습니다. 그리고 나서 M번의 명령을 통해 로봇들의 안정성을 검증할 수 있는데 move() 메서드를 이용해 움직일 로봇을 찾고, 명령에 따라 회전하거나 이동시켜 주었습니다. 만약 로봇이 움직였을 때,  그다음 움직일 위치가 벽에 충돌하는 경우 또는 다른 로봇과 충돌하는 경우에는 그 충돌을 출력해주고 종료해주었습니다. 반대로 로봇이 잘 움직였다면 그 마지막 위치의 정보를 다시 list에 넣어주고 다음 명령을 반복했습니다. M번의 로봇이 정상적으로 명령을 수행했다면 OK를 출력하고 종료해주었습니다.

코드

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
import java.util.ArrayList;
import java.util.StringTokenizer;
 
public class Main_BJ_2174_로봇시뮬레이션 {
    
    static class robot{
        int num ,x, y, dir;
 
        public robot(int num, int x, int y, int dir) {
            super();
            this.num = num;
            this.x = x;
            this.y = y;
            this.dir = dir;
        }
    }
    
    static int A,B,N,M;
    static int [][] map;
    static int [] dx = {-1,0,1,0};
    static int [] dy = {0,1,0,-1};
     
    static ArrayList<robot> list = new ArrayList<>();
    
    public static void main(String[] args) throws IOException {
        BufferedReader br = new BufferedReader(new InputStreamReader(System.in));
        StringTokenizer st;
        
        st = new StringTokenizer(br.readLine());
        
        A = Integer.parseInt(st.nextToken()); //가로
        B = Integer.parseInt(st.nextToken()); //세로
        
        map = new int[B][A];
        
        st = new StringTokenizer(br.readLine());
        N = Integer.parseInt(st.nextToken()); //로봇수 
        M = Integer.parseInt(st.nextToken()); // 명령수
        
        int num=0;
        for(int i=1;i<=N;i++) {
            st = new StringTokenizer(br.readLine());
            int x = Integer.parseInt(st.nextToken())-1;
            int y = Integer.parseInt(st.nextToken())-1;
            String dir = st.nextToken();
            
            map[B-y-1][x] = i;
            if(dir.equals("N")) num=0;
            else if(dir.equals("E")) num=1;
            else if(dir.equals("S")) num=2;
            else if(dir.equals("W")) num=3;
            
            list.add(new robot(i,B-y-1,x,num));
        }
        
        for(int i=0;i<M;i++) {
            st = new StringTokenizer(br.readLine());
            int robot = Integer.parseInt(st.nextToken());
            String command = st.nextToken();
            int repetition = Integer.parseInt(st.nextToken());
            move(robot,command,repetition);    
        }
        
        System.out.println("OK");
 
    }
    
    public static void move(int robot, String command, int repetition) {
        
        for(int i=0;i<list.size();i++) {
            if(list.get(i).num==robot) { //움직일 로봇 찾고
                int direction = list.get(i).dir;
                if(command.equals("L")) { //왼쪽으로 90도
                    for(int j=0;j<repetition;j++) {
                        if(direction==0) direction=3;
                        else direction--;
                    }
                    list.set(i, new robot(list.get(i).num,list.get(i).x,list.get(i).y,direction));
                }else if(command.equals("R")) { //오른쪽으로 90도
                    for(int j=0;j<repetition;j++) {
                        if(direction==3) direction=0;
                        else direction++;
                    }
                    list.set(i, new robot(list.get(i).num,list.get(i).x,list.get(i).y,direction));
                }else if(command.equals("F")) { //현재 방향에서 한칸 앞으로
                    int x = list.get(i).x;
                    int y = list.get(i).y;
                    map[x][y] = 0;
                    int nx=0, ny=0;
                    for(int j=0;j<repetition;j++) {
                        nx = x+dx[direction];
                        ny = y+dy[direction];
                        
                        if(range(nx,ny)) {
                            if(map[nx][ny]==0) {
                                x = nx; y = ny;
                            }else {
                                System.out.println("Robot "+list.get(i).num+" crashes into robot "+map[nx][ny]);
                                System.exit(0);
                            }
                        }else {
                            System.out.println("Robot "+list.get(i).num+" crashes into the wall");
                            System.exit(0);
                        }
                    }
                    map[nx][ny] = list.get(i).num;
                    list.set(i, new robot(list.get(i).num,nx,ny,direction));
                }
            }
        }
        
    }
    
    public static boolean range(int x, int y) {
        return x>=0 && x<&& y>=0 && y<A;
    }
 
}
cs

 

 

문제

N명의 학생들을 키 순서대로 줄을 세우려고 한다. 각 학생의 키를 직접 재서 정렬하면 간단하겠지만, 마땅한 방법이 없어서 두 학생의 키를 비교하는 방법을 사용하기로 하였다. 그나마도 모든 학생들을 다 비교해 본 것이 아니고, 일부 학생들의 키만을 비교해 보았다.

일부 학생들의 키를 비교한 결과가 주어졌을 때, 줄을 세우는 프로그램을 작성하시오.

 

입력

첫째 줄에 N(1 ≤ N ≤ 32,000), M(1 ≤ M ≤ 100,000)이 주어진다. M은 키를 비교한 회수이다. 다음 M개의 줄에는 키를 비교한 두 학생의 번호 A, B가 주어진다. 이는 학생 A가 학생 B의 앞에 서야 한다는 의미이다.

학생들의 번호는 1번부터 N번이다.

 

출력

첫째 줄에 학생들을 키 순서대로 줄을 세운 결과를 출력한다. 답이 여러 가지인 경우에는 아무거나 출력한다.

 

예제

 

 

코드

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
import java.util.ArrayList;
import java.util.LinkedList;
import java.util.Queue;
import java.util.StringTokenizer;
 
public class Main_BJ_2252_줄세우기 {
    
    
    static int N,M;
    static ArrayList<ArrayList<Integer>> list = new ArrayList<ArrayList<Integer>>();
    static int [] indegree;
 
    public static void main(String[] args) throws IOException {
        BufferedReader br = new BufferedReader(new InputStreamReader(System.in));
        StringTokenizer st;
        
        st = new StringTokenizer(br.readLine());
        N = Integer.parseInt(st.nextToken());
        M = Integer.parseInt(st.nextToken());
        
        indegree = new int[N+1];
        
        for(int i=0;i<=N;i++) list.add(new ArrayList<>());
        
        for(int i=0;i<M;i++) {
            st = new StringTokenizer(br.readLine());
            
            int a = Integer.parseInt(st.nextToken());
            int b = Integer.parseInt(st.nextToken());
            
            list.get(a).add(b);
            indegree[b]++;
        }
        sort();
    }
    
    public static void sort() {
        Queue<Integer> queue = new LinkedList<>();
        Queue<Integer> result = new LinkedList<>();
        
        for(int i=1;i<N+1;i++) {
            if(indegree[i]==0) {
                queue.add(i);
            }
        }
        
        while(!queue.isEmpty()) {
            int temp = queue.poll();
            result.add(temp);
            
            for(Integer item : list.get(temp)) {
                indegree[item]--;
                if(indegree[item]==0) {
                    queue.add(item);
                }
            }
        }
        while(!result.isEmpty()) {
            System.out.print(result.poll()+" ");
        }
    }
 
}
cs

 

'문제 > 백준' 카테고리의 다른 글

백준 1043번 거짓말 [JAVA]  (0) 2021.06.25
백준 2174번 로봇 시뮬레이션 [JAVA]  (0) 2021.06.25
백준 10282번 해킹 [JAVA]  (0) 2021.06.17
백준 19942 다이어트 [JAVA]  (0) 2021.06.16
백준 5014번 스타트링크 [JAVA]  (0) 2021.06.16

문제

최흉최악의 해커 yum3이 네트워크 시설의 한 컴퓨터를 해킹했다! 이제 서로에 의존하는 컴퓨터들은 점차 하나둘 전염되기 시작한다. 어떤 컴퓨터 a가 다른 컴퓨터 b에 의존한다면, b가 감염되면 그로부터 일정 시간 뒤 a도 감염되고 만다. 이때 b가 a를 의존하지 않는다면, a가 감염되더라도 b는 안전하다.

최흉최악의 해커 yum3이 해킹한 컴퓨터 번호와 각 의존성이 주어질 때, 해킹당한 컴퓨터까지 포함하여 총 몇 대의 컴퓨터가 감염되며 그에 걸리는 시간이 얼마인지 구하는 프로그램을 작성하시오.

 

입력

첫째 줄에 테스트 케이스의 개수가 주어진다. 테스트 케이스의 개수는 최대 100개이다. 각 테스트 케이스는 다음과 같이 이루어져 있다.

  • 첫째 줄에 컴퓨터 개수 n, 의존성 개수 d, 해킹당한 컴퓨터의 번호 c가 주어진다(1 ≤ n ≤ 10,000, 1 ≤ d ≤ 100,000, 1 ≤ c ≤ n).
  • 이어서 d개의 줄에 각 의존성을 나타내는 정수 a, b, s가 주어진다(1 ≤ a, b ≤ n, a ≠ b, 0 ≤ s ≤ 1,000). 이는 컴퓨터 a가 컴퓨터 b를 의존하며, 컴퓨터 b가 감염되면 s초 후 컴퓨터 a도 감염됨을 뜻한다.

각 테스트 케이스에서 같은 의존성 (a, b)가 두 번 이상 존재하지 않는다.

 

출력

각 테스트 케이스마다 한 줄에 걸쳐 총 감염되는 컴퓨터 수, 마지막 컴퓨터가 감염되기까지 걸리는 시간을 공백으로 구분지어 출력한다.

 

예제

 

풀이 방법

해커로부터 컴퓨터를 해킹해 서로 의존하는 컴퓨터들이 모두 감염되는 시간과, 감염된 컴퓨터의 수를 구하는 문제였습니다.

 

저는 배열과 리스트 두 가지 방식으로 다익스트라 알고리즘을 이용해 문제를 해결했습니다.

 

1) 연결되어 있는 컴퓨터 정보 배열/리스트에 저장

2) 컴퓨터간의 감염시간을 저장할 distance 배열 큰 수로 초기화

3) 다익스트라 알고리즘을 이용해 최소 감염시간 distance배열에 저장

4) 감염되는 시간 및 감염된 컴퓨터 수를 구하기 위해 distance배열 값이 INF가 아닌 경우 count++ 및 distance 최댓값 구하기

5) count 및 최대 distance 값 출력

 

코드 (배열)

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;
import java.util.StringTokenizer;
 
public class Main_BJ_10282_해킹_array {
    
    static class info{
        int end, time;
 
        public info(int end, int time) {
            super();
            this.end = end;
            this.time = time;
        }
 
    }
    
    static int INF = Integer.MAX_VALUE;
    static boolean[] visited;
    static List<info> list[];
    static int [] distance;
 
    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());
        
        for(int tc=0; tc<T;tc++) {
            st = new StringTokenizer(br.readLine());
            int n = Integer.parseInt(st.nextToken());
            int d = Integer.parseInt(st.nextToken());
            int c = Integer.parseInt(st.nextToken());
            
            list = new ArrayList[n+1];
            distance = new int[n+1];
            visited = new boolean[n+1];
            
            for(int i=1;i<=n;i++) list[i] = new ArrayList<>();
            
            for(int i=0;i<d;i++) {
                st = new StringTokenizer(br.readLine());
                int a = Integer.parseInt(st.nextToken());
                int b = Integer.parseInt(st.nextToken());
                int s = Integer.parseInt(st.nextToken());
                
                list[b].add(new info(a,s));
            }
            
            Arrays.fill(distance, INF);
            distance[c]=0;
            
            int min, current = 0;
            for(int i=0; i<n;i++) {
                min = Integer.MAX_VALUE;
                current = -1;
                for(int j=1;j<n+1;j++) {
                    if(!visited[j] && distance[j]<min) {
                        min = distance[j];
                        current =j;
                    }
                }
                if(current == -1break;
                
                for(info next : list[current]) {
                    if(!visited[next.end] && distance[next.end]> distance[current]+next.time) {
                        distance[next.end] = distance[current] + next.time;
                    }
                }
                visited[current] = true;
            }
            
            int count=0, max=0;
            for(int i=1;i<distance.length;i++) {
                if(distance[i]!=INF) {
                    count++;
                    max = Math.max(max, distance[i]);
                }
                
            }
            sb.append(count+" "+max+"\n");
        }
        System.out.println(sb.toString());
    }
 
}
cs

 

코드 (리스트)

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Comparator;
import java.util.PriorityQueue;
import java.util.StringTokenizer;
 
public class Main_BJ_10282_해킹_list {
    
    static class info{
        int end, time;
 
        public info(int end, int time) {
            super();
            this.end = end;
            this.time = time;
        }
 
    }
    
    static int INF = Integer.MAX_VALUE;
    static boolean[] visited;
    static ArrayList<ArrayList<info>> list;
    static int [] distance;
    static PriorityQueue<info> pq = new PriorityQueue<>(new Comparator<info>() {
 
        @Override
        public int compare(info o1, info o2) {
            return o1.time-o2.time;
        }
    });
 
    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());
        
        for(int tc=0; tc<T;tc++) {
            st = new StringTokenizer(br.readLine());
            int n = Integer.parseInt(st.nextToken());
            int d = Integer.parseInt(st.nextToken());
            int c = Integer.parseInt(st.nextToken());
            
            list = new ArrayList<>();
            distance = new int[n+1];
            visited = new boolean[n+1];
            
            for(int i=0;i<=n;i++) list.add(new ArrayList<>());
            
            for(int i=0;i<d;i++) {
                st = new StringTokenizer(br.readLine());
                int a = Integer.parseInt(st.nextToken());
                int b = Integer.parseInt(st.nextToken());
                int s = Integer.parseInt(st.nextToken());
                
                list.get(b).add(new info(a,s));
            }
            
            Arrays.fill(distance, INF);
            
            distance[c]=0;
            
            pq.add(new info(c,0));
            while(!pq.isEmpty()) {
                info temp = pq.poll();
                if(visited[temp.end]) continue;
                visited[temp.end]= true;
                
                for(info node : list.get(temp.end)) {
                    if(distance[node.end]>distance[temp.end]+node.time) {
                        distance[node.end] = distance[temp.end]+node.time;
                        pq.add(new info(node.end,distance[node.end]));
                    }
                }
            }
            
            
            int count=0, max=0;
            for(int i=1;i<distance.length;i++) {
                if(distance[i]!=INF) {
                    count++;
                    max = Math.max(max, distance[i]);
                }
                
            }
            sb.append(count+" "+max+"\n");
        }
        System.out.println(sb.toString());
    }
 
}
cs

 

 

문제

식재료 N개 중에서 몇 개를 선택해서 이들의 영양분(단백질, 탄수화물, 지방, 비타민)이 일정 이상이 되어야 한다. 아래 표에 제시된 6가지의 식재료 중에서 몇 개를 선택해서 이들의 영양분의 각각 합이 최소 100, 70, 90, 10가 되도록 하는 경우를 생각해보자. 이 경우 모든 재료를 선택하면 쉽게 해결되지만, 우리는 조건을 만족시키면서도 비용이 최소가 되는 선택을 하려고 한다.

재료 단백질 지방 탄수화물 비타민 가격

1 30 55 10 8 100
2 60 10 10 2 70
3 10 80 50 0 50
4 40 30 30 8 60
5 60 10 70 2 120
6 20 70 50 4 40

예를 들어, 식재료 1, 3, 5를 선택하면 영양분은 100, 145, 130, 10으로 조건을 만족하지만 가격은 270이 된다. 대신 2, 3, 4를 선택하면 영양분의 합은 110, 130, 90, 10, 비용은 180이 되므로, 앞의 방법보다는 더 나은 선택이 된다.

입력으로 식재료 표가 주어졌을 때, 최저 영양소 기준을 만족하는 최소 비용의 식재료 집합을 찾아야 한다.

 

입력

첫 줄에 식재료의 개수 N이 주어진다.

다음 줄에는 단백질, 지방, 탄수화물, 비타민의 최소 영양성분을 나타내는 정수 mp, mf, ms, mv가 주어진다.

이어지는 N개의 각 줄에는 i번째 식재료의 단백질, 지방, 탄수화물, 비타민과 가격이 5개의 정수 pi, fi, si, vi, ci와 같이 주어진다. 식재료의 번호는 1부터 시작한다.

 

출력

첫 번째 줄에 최소 비용을 출력하고, 두 번째 줄에 조건을 만족하는 최소 비용 식재료의 번호를 공백으로 구분해 오름차순으로 한 줄에 출력한다. 같은 비용의 집합이 하나 이상이면 사전 순으로 가장 빠른 것을 출력한다.

조건을 만족하는 답이 없다면 -1을 출력하고, 둘째 줄에 아무것도 출력하지 않는다.

 

예제

 

풀이 방법

선택한 식재료들의 영양분 합이 최저 영양소 기준을 만족하면서 최소 비용의 식재료의 집합을 찾는 문제였습니다.

 

일단 먼저, 만족하는 집합을 찾기 위해 식재료를 최소 1개 이상 최대 N개 이하를 선택할 수 있기 때문에 반복문을 통해 해당 식재료 개수만큼 선택하여 그 선택된 식재료의 영양분 합이 최저 영양소의 기준을 만족하나 확인해 주었습니다.

isCheck() 메서드에서 선택한 식재료들의 영양분의 합들을 sum배열에 넣어준 뒤, 하나라도 최저 영양소 기준을 만족하지 못한다면 false를 리턴해주고 이외에는 선택된 식재료들의 가격을 비교하여 최소 비용의 식재료의 집합을 찾아주었습니다. 

 

출력 조건이 같은 비용의 집합이 하나 이상이면 사전 순으로 가장 빠른 것을 출력해야 하기 때문에, list에 넣어 정렬을 한 뒤 가장 첫 번째 리스트의 값을 출력해주었습니다. 만약 조건을 만족하는 집합이 없다면 -1을 출력해주었습니다. 

코드

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
import java.util.ArrayList;
import java.util.Collections;
import java.util.StringTokenizer;
 
public class Main_BJ_19942_다이어트 {
    
    static int N, M=5,ans =Integer.MAX_VALUE;
    static int [][] nutrients;
    static int [] minN;
    static int []select;
    
    static ArrayList<String> list = new ArrayList<>();
    
    public static void main(String[] args) throws NumberFormatException, IOException {
        BufferedReader br = new BufferedReader(new InputStreamReader(System.in));
        StringTokenizer st;
        
        N = Integer.parseInt(br.readLine());
        
        nutrients = new int[N][M]; // N개의 영양분을 담을 배열
        minN = new int[4]; // 최저 영양소 기준을 담을 배열
        
        st = new StringTokenizer(br.readLine());
        for(int i=0;i<4;i++) minN[i] = Integer.parseInt(st.nextToken());
        
        for(int i=0;i<N;i++) {
            st = new StringTokenizer(br.readLine());
            for(int j=0;j<M;j++) {
                nutrients[i][j] = Integer.parseInt(st.nextToken());
            }
        }
        for(int i=1;i<=N;i++) {
            select = new int[N];
            choice(0,i,0);
        }
        if(list.size()>0) {
            System.out.println(ans);
            Collections.sort(list);
            String st1 = list.get(0);
            for(int i=0;i<st1.length();i++) {
                System.out.print(st1.charAt(i));
            }
        }else System.out.println(-1);
    }
    
    public static void choice(int cnt, int sel, int start) {
        if(cnt==sel) {
           isCheck(sel); //조건체크
            return;
        }
        for(int i=start;i<N;i++) {
                select[cnt]=i;
                choice(cnt+1,sel,i+1);
        }
        
    }
    
    public static boolean isCheck(int sel) {
        int price=0;
        int []sum = new int[4];
        for(int i=0;i<sel;i++) {
                sum[0]+=nutrients[select[i]][0];
                sum[1]+=nutrients[select[i]][1];
                sum[2]+=nutrients[select[i]][2];
                sum[3]+=nutrients[select[i]][3];
                price+=nutrients[select[i]][4];            
        }
        
        for(int i=0;i<4;i++) {
            if(minN[i]>sum[i]) return false;
        }
        
        if(ans>=price) {
            if(ans>price) {
                list.clear();
            }
            String str="";
            for(int i=0;i<sel;i++) {
                str+=(select[i]+1+" ");
            }
            list.add(str);
            ans = price;
        }
        return true;
    }
}
cs

 

'문제 > 백준' 카테고리의 다른 글

백준 2252번 줄세우기 [JAVA]  (0) 2021.06.18
백준 10282번 해킹 [JAVA]  (0) 2021.06.17
백준 5014번 스타트링크 [JAVA]  (0) 2021.06.16
백준 9207번 페그 솔리테어 [JAVA]  (0) 2021.06.15
백준 17143번 낚시왕 [JAVA]  (0) 2021.06.15

+ Recent posts