Problem


주어지는 두개의 지도를 겹친 결과를 출력하는 문제입니다.



Solution

그림만 봐도 알수 있듯이 겹치는 부분은 # 겹치지 않은 부분은 빈칸으로 나옵니다.


친절하게 비트로 풀라고 안내까지 해주고 있으니 Integer.toBinaryString 함수를 사용하여 계산을 해줍니다.


1 과 0 으로 이루어진 String을 replaceAll로 전부 바꿔주면 됩니다.


한 가지 주의할 점이 하나 있습니다.


1111(2) 처럼 자릿수가 작은 글자는 n이 5일때 01111로 바꿔주어야 합니다.


여기서는 자리수를 맞춰주는 String.format 함수를 사용하여 n 자리만큼 공백으로 채워줍니다.



Java Code

class Solution {
    static public String[] solution(int n, int[] arr1, int[] arr2) {
        String[] answer = new String[n];

        for (int i=0; i<n; i++) {          
            String temp = Integer.toBinaryString(arr1[i] | arr2[i]);  
            temp = String.format("%" + n + "s", temp);
            temp = temp.replaceAll("1", "#");
            temp = temp.replaceAll("0", " ");
            answer[i] = temp;
        }

        return answer;
    }
}


문제 링크 : https://www.acmicpc.net/problem/2811


해석 때문인지는 몰라도 문제를 이해하는데 살짝 어려움을 겪은 문제입니다.

 

결론을 말하자면 상범이에게 꽃을 주는 날의 최대값을 구하면 됩니다.

 

이 문제의 핵심은 다음과 같습니다.

 

1. 상범이의 우울한 정도는 상관없다. 양수, 음수인지만 판단

2. 꽃은 하루에 한개씩만 준다.

3. 꽃을 다 주지 못해도 최대한 준다. 둘째날이 우울한 날이라면 꽃은 첫째날밖에 못준다. 그래도 주자.

4. 3T만큼 꽃을 주는 날은 여러 개의 최장 우울 기간 중 한번이다.

 

문제를 해결하려면 결국 최장 우울 기간 중 가장 꽃을 많이 줄 수 있는 날이 언제인지 찾는 것이 중요합니다.

 

이걸 직접 구하는 것보다는 최장 우울 기간 전부 탐색하면서 max값을 구하는 것이 편합니다.

 

우선 모든 날에 대해서 2T만큼 꽃 주는 날을 구합니다.

 

그 후에 최장 우울 기간들만 스캔하면서 또다시 3T만큼 카운트합니다.

 

이 때 2T 구할때 체크했던 날들은 제외해서 중복 체크를 방지합니다.

 

문제에 있는 예제로 간단하게 설명해보겠습니다.

 

먼저 상범이의 우울한 날을 구합니다.

 

그림 1. 상범이의 우울한 날

그림 1에서 상범이의 최장 우울 기간은 1이고, 최장 우울 기간의 갯수는 3개입니다.

 

그럼 우선 최장 우울 기간과 관계 없이 모든 우울 기간에 대해서 2T만큼 꽃을 주겠습니다.

 

 

그림 2. 첫번째 우울한 날에 대해 꽃 주기

우울한 날이 2일이므로 0일과 1일에 꽃을 주어야 합니다.

 

하지만 보시다시피 0일은 존재하지 않습니다.

 

이 때, 위에서 언급한 핵심 3번을 보시면 꽃을 다 주지 못해도 최대한 주라는 조건이 있습니다.

 

그래서 그림 2처럼 체크하면 됩니다.

 

 

그림 3. 두번째 우울한 날에 대해 꽃 주기

그리고 두번째로 우울한 날에 대해서 꽃을 줍니다.

 

그림 3처럼 4일과 5일에 꽃을 줄 수 있습니다.

 

 

 

그림 4. 세번째 우울한 날에 대해 꽃 주기

마지막으로 우울한 날에 대해서 꽃을 줍니다.

 

8일이 우울한 날이기 때문에 6일과 7일에 꽃을 줍니다.

 

여기서 헷갈릴 수 있는 요소가 하나 있는데 우울한 날과 관계없이 꽃은 줄 수 있습니다.

 

따라서 그림 4처럼 2T만큼 모든 우울한 날에 대해서 체크를 하였습니다.

 

2T를 전부 체크한 시점에서 꽃을 주는 날의 count는 5 입니다.

 

 

 

그림 4. 3T 구하기

이제 3T의 최대값을 구해야 합니다.

 

최장 우울 기간에 해당하는 모든 우울한 날에 대해서 3T를 구합니다.

 

첫번째 날은 보시다시피 꽃을 더 줄 수 없습니다.

 

add는 0 입니다.

 

 

 

그림 5. 3T 구하기 - 2

두번째 우울한 날에 대해서 3T를 구합니다.

 

우울 기간이 1이기 때문에 꽃을 하루 더 줄 수 있습니다.

 

이때, 꽃을 줄 수 있는 날은 카운트만 하고 표시는 하지 않습니다.

 

3T는 모든 여러가지 최장 우울 기간 중 단 한번만 주면 되기 때문입니다.

 

만약 최장 우울 기간이 1이 아니라 3 정도 되고 3T를 구할때마다 모든 날에 체크를 한다면 뒤에 나오는

 

최장 우울 기간에는 꽃을 줄  수 있는 날이 없을수도 있고, 그럼 모든 경우의 수를 알 수 없습니다.

 

아무튼 여기서 add는 1이고 기존의 count 5와 더해서 max 값을 6으로 갱신합니다.

 

 

그림 6. 3T 구하기 - 3

마지막 우울한 날에 대해서 3T를 구합니다.

 

하지만 이미 3일 전인 5, 6, 7일 전부 꽃을 주기로 되있으므로 꽃을 더 줄 필요가 없습니다.

 

따라서 추가로 주는 꽃은 없고 add는 0입니다.

 

 

 

그림 7. 최종

최종적으로 꽃을 주는 날은 위와 같고 최대값은 6이 됩니다.


** 코드에서는 뒤에서부터 돌면서 미리 꽃을 주었습니다.




문제 링크 : https://www.acmicpc.net/problem/17284


매우 간단한 구현 문제입니다.


종료 조건이 없기 때문에 StringTokenizer의 st.hasMoreTokens로 토큰이 있는지 검사해도 되지만


저는 어차피 첫번째 줄에만 입력이 들어온다는 조건이 있기 때문에 한줄 받아서 Split(" ")을 사용했습니다.



출처 : https://gist.github.com/psayre23/c30a821239f4818b0709

 

Runtime Complexity of Java Collections

Runtime Complexity of Java Collections. GitHub Gist: instantly share code, notes, and snippets.

gist.github.com

 

'알고리즘 문제 > 공부' 카테고리의 다른 글

2차원 배열 회전 Rotate (Java)  (1) 2020.03.21
자바 출력 (Java)  (0) 2019.01.05
자바 입력 (Java)  (0) 2019.01.05
위상정렬 Topological Sort (Java)  (1) 2018.12.28
부분집합 PowerSet (Java)  (2) 2018.12.27

문제 링크 : https://www.acmicpc.net/problem/6118

 

6118번: 숨바꼭질

문제 재서기는 수혀니와 교외 농장에서 숨바꼭질을 하고 있다. 농장에는 헛간이 많이 널려있고 재석이는 그 중에 하나에 숨어야 한다. 헛간의 개수는 N(2 <= N <= 20,000)개이며, 1 부터 샌다고 하자.   재서기는 수혀니가 1번 헛간부터 찾을 것을 알고 있다. 모든 헛간은 M(1<= M <= 50,000)개의 양방향 길로 이어져 있고, 그 양 끝을 A_i 와 B_i(1<= A_i <= N; 1 <= B_i <= N; A_i != B_i)로 나타낸

www.acmicpc.net

수혀니가 있는 1번 헛간으로부터 가장 먼 헛간들의 갯수, 거리를 구하는 문제입니다.

 

간선 가중치가 없고 모든 최장거리를 전부 구해야되기 때문에 BFS를 사용하면 쉽게 해결할 수 있습니다.

 

한가지 주의할 점은 N이 최대 20,000 이기 때문에 O(N^2)인 인접행렬을 이용할 경우 시간초과가 납니다.

 

인접리스트 BFS를 돌면서 최대 거리, 최대 거리의 갯수, 재서기가 숨은 가장 작은 헛간을 찾아주시면 됩니다.

 

문제 링크 : https://www.acmicpc.net/problem/1620


문제는 굉장히 길지만 핵심은 포켓몬을 말하면 번호를, 번호를 말하면 포켓몬을 출력하는 문제입니다.


포켓몬이 최대 100,000마리이기 때문에 Map 자료구조를 사용하여 key, value를 바꿔가며 두번씩 넣었습니다.


간단하니 코드로 보는게 이해하기 쉽습니다.


문제 링크 : https://www.acmicpc.net/problem/1697


수빈이가 동생과 만나는 시간을 출력하는 문제입니다.


간단하게 BFS를 이용하여 해결할 수 있습니다.


처음 수빈이의 위치를 Queue에 넣고 +1, -1, *2 한 위치를 각각 Queue에 넣어가며 BFS를 합니다.


그리고 수빈이가 한번 방문했던 위치를 계속 체크하여 중복 방문을 하지 않도록 하면 됩니다.


그리고 수빈이의 시작 위치가 동생보다 크다면 무조건 -1을 반복하여 만나는 수밖에 없으므로 N-K를 바로 출력합니다


문제 링크 : https://www.acmicpc.net/problem/1238


모든 마을에서 출발하여 X 마을에 들렀다가 다시 돌아오는 길이 가장 큰 값을 구하는 문제입니다.


N이 크지 않아서 플로이드와샬, 다익스트라 어느것으로 해도 풀리는 문제입니다.


하지만 X에 대한 최단 경로만 구하면 되는 상황에서 모든 정점에 대해 다익스트라를 하는 것은 비효율 적입니다.


그래서 'X마을 -> 다른 모든 마을' 의 최단경로와 '다른 모든 마을 -> X마을' 의 최단경로.


단 두번만 다익스트라를 돌려서 문제를 해결할 수 있습니다.


X마을에서의 최단 경로는 시작점을 X로 놓고 하면 되지만


X마을로 향하는 최단 경로는 조금 다릅니다.


입력받는 간선의 값을 전부 뒤집어서 넣은 뒤에 시작점을 X로 하여 구해야 합니다.


인접행렬을 예로 들면 원래는 edge[v1][v2] = cost 이렇게 입력하는 식을 edge[v2][v1] = cost 로 입력한 edge 배열을 이용하여 구할 수 있습니다.


저는 처음부터 인접리스트를 두개 생성하여 구하는 방법으로 하였습니다.


+ Recent posts