• 생각

1. 코테보면 나올만한 문제인 것 같다. 다음과 같은 조건을 주의해서 코드를 짜면 될 것 같다.

 

조건 ) 1. <문자열>은 뒤집지 않고 출력 2. 문자열'띄어쓰기' or 문자열'문자열끝' 문자열 부분을 뒤집어서 출력

 

2. 스택을 잘 사용하면 시간이 더 빠르게 나올 것 같다.

 

3. StringTokenizer - 띄어쓰기된 문자열별로 .nextToken()으로 받아와서 작업해도 더 빠를 것 같다.

 

  • 코드

정답 코드 : 위의 조건들을 고려하여서 코드를 짰다.

 

import java.io.BufferedReader;
import java.io.InputStreamReader;
import java.util.LinkedList;
import java.util.Queue;
import java.util.StringTokenizer;
import java.util.regex.Pattern;

public class Main {

	public static void main(String[] args) throws Exception {
		BufferedReader br = new BufferedReader(new InputStreamReader(System.in));
		String s = br.readLine();
		String temp = "";
		StringBuilder sb = new StringBuilder();

		boolean check = false;
		for(int i = 0; i < s.length(); i++) {
			if(s.charAt(i) == '<')		// 뒤집지 않는 시작 문자 
				check = true;
			
			if(check)			// 뒤집지 않아야할 문자열
				temp = temp + s.charAt(i);
			else {				// 뒤집어야할 문자열
				if(s.charAt(i) == ' ')
					temp = temp + s.charAt(i);
				else
					temp = s.charAt(i) + temp;
			}
			
			if(s.charAt(i) == '>')		// 뒤집지 않는 끝 문자
				check = false;
			
			// ' ' or '>' or 문자열 끝이면 append
			if(s.charAt(i) == '>' || s.charAt(i) == ' ' || i == s.length()-1) {
				sb.append(temp);
				temp = "";
			}
			
		}
		System.out.println(sb);
	}
}

 

 

다른 사람 코드 : 스택으로 구현한 코드.

 

import java.io.BufferedReader;
import java.io.BufferedWriter;
import java.io.IOException;
import java.io.InputStreamReader;
import java.io.OutputStreamWriter;
import java.util.Stack;

public class Main {
    public static final char START_TAG = '<';
    public static final char END_TAG = '>';
    public static final char SPACE = ' ';

    public static void main(String args[]) {
        try (BufferedReader br = new BufferedReader(new InputStreamReader(System.in));
                BufferedWriter bw = new BufferedWriter(new OutputStreamWriter(System.out));) {
            String line = br.readLine();
            StringBuffer result = new StringBuffer();
            Stack<Character> word = new Stack<>();
            Boolean isTag = Boolean.FALSE;
            for (char c : line.toCharArray()) {
                if (c == START_TAG) {
                    if (!word.isEmpty()) {
                        result.append(reverse(word));
                    }
                    isTag = Boolean.TRUE;
                    result.append(c);
                } else if (c == END_TAG) {
                    result.append(c);
                    isTag = Boolean.FALSE;
                } else if (c == SPACE) {
                    if (isTag) {
                        result.append(c);
                    } else {
                        result.append(reverse(word));
                        result.append(SPACE);
                    }
                } else {
                    if (isTag) {
                        result.append(c);
                    } else {
                        word.add(c);
                    }
                }
            }
            if (!word.isEmpty()) {
                result.append(reverse(word));
            }
            bw.write(result.toString() + System.lineSeparator());

        } catch (IOException ioe) {
            ioe.printStackTrace();
        }
    }

    static String reverse(Stack<Character> input) {
        StringBuffer sb = new StringBuffer();
        while (!input.empty()) {
            sb.append(input.pop());
        }
        return sb.toString();
    }
}

 

다른 사람 코드 : StringTokenizer를 통해 구현한 코드가 있었다. 내가 생각했던 방식보다 더 활용을 잘한 코드이다. 제일 빨랐던 코드인만큼 공부하면 좋을 것 같다.

 

import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
import java.util.StringTokenizer;

public class Main {
	static BufferedReader reader = new BufferedReader(new InputStreamReader(System.in));

	static void solve(String target) {
		StringBuilder builder = new StringBuilder();
		StringTokenizer leftTagTokenizer = new StringTokenizer(target, "<");
		while (leftTagTokenizer.hasMoreTokens()) {
			String leftToken = leftTagTokenizer.nextToken();
			boolean includeTag = leftToken.contains(">");
			StringTokenizer rightTagTokenizer = new StringTokenizer(leftToken, ">");
			while (rightTagTokenizer.hasMoreTokens()) {
				if (includeTag) {
					builder.append("<");
					builder.append(rightTagTokenizer.nextToken());
					builder.append(">");
					includeTag = false;
				} else {
					String tagToken = rightTagTokenizer.nextToken();
					StringTokenizer spaceTokenizer = new StringTokenizer(tagToken);
					while (spaceTokenizer.hasMoreTokens()) {
						StringBuilder reverseBuilder = new StringBuilder(spaceTokenizer.nextToken());
						reverseBuilder.reverse();
						builder.append(reverseBuilder);
						if (spaceTokenizer.hasMoreTokens()) {
							builder.append(" ");
						}
					}
				}
			}
		}
		System.out.println(builder);
	}

	public static void main(String[] args) throws IOException {
		String line = reader.readLine();
		solve(line);
	}
}

 

 


 

  • 생각

1. queue를 두개를 만들어서 고슴도치가 움직이는 queue, 물이 차는 queue를 만들어서 bfs를 돈다. 이게 최선일 것 같은느낌..?

 

  • 코드

정답 코드 : 기본적인 BFS 문제인 것 같다. 중간에 코드를 잘 못 적어서 1시간 삽질했다..

 

import java.io.BufferedReader;
import java.io.InputStreamReader;
import java.util.LinkedList;
import java.util.Queue;
import java.util.StringTokenizer;

public class Main {
	static int N, M, minValue, destinationX, destinationY;
	static Queue<int[]> HedgehogQueue, waterQueue;
	static char[][] array;
	static boolean[][] check;
	static int[] x = { -1, 1, 0, 0 };
	static int[] y = { 0, 0, -1, 1 };

	public static void main(String[] args) throws Exception {
		SetData();
		bfs();
		if (minValue == Integer.MAX_VALUE)
			System.out.println("KAKTUS");
		else
			System.out.println(minValue);
	}

	private static void SetData() throws Exception {
		BufferedReader br = new BufferedReader(new InputStreamReader(System.in));
		StringTokenizer st = new StringTokenizer(br.readLine());

		N = Integer.parseInt(st.nextToken());
		M = Integer.parseInt(st.nextToken());
		array = new char[N][M];
		check = new boolean[N][M];
		HedgehogQueue = new LinkedList<>();
		waterQueue = new LinkedList<>();
		minValue = Integer.MAX_VALUE;
		String s;

		for (int i = 0; i < N; i++) {
			s = br.readLine();
			for (int j = 0; j < M; j++) {
				array[i][j] = s.charAt(j);
				if (array[i][j] == 'D') {
					destinationX = i;
					destinationY = j;
				} else if (array[i][j] == 'S') {
					HedgehogQueue.offer(new int[] { i, j, 0 });
					check[i][j] = true;
				} else if (array[i][j] == '*') {
					waterQueue.offer(new int[] { i, j });
				}
			}
		}
	}

	private static void bfs() {
		while (!HedgehogQueue.isEmpty()) {
			int size = waterQueue.size();
			for (int i = 0; i < size; i++) {
				int location[] = waterQueue.poll();
				for (int direction = 0; direction < 4; direction++) {
					int r = location[0] + x[direction]; // 물 방향
					int c = location[1] + y[direction];

					if (r < 0 || r >= N || c < 0 || c >= M)
						continue;
					if (array[r][c] != '.')
						continue;
					array[r][c] = '*';
					waterQueue.offer(new int[] { r, c });
				}
			}

			size = HedgehogQueue.size();
			for (int i = 0; i < size; i++) {
				int location[] = HedgehogQueue.poll();
				int count = location[2];

				if (location[0] == destinationX && location[1] == destinationY) {
					minValue = Math.min(minValue, count); // 마지막 위치에 도달했을 때 지나온 칸 수를 return
				}

				for (int direction = 0; direction < 4; direction++) {
					int r = location[0] + x[direction]; // 고슴도치 방향
					int c = location[1] + y[direction];

					if (r < 0 || r >= N || c < 0 || c >= M)
						continue;
					if (check[r][c] || array[r][c] == '*' || array[r][c] == 'X')
						continue;
					check[r][c] = true;
					HedgehogQueue.offer(new int[] { r, c, count + 1 });
				}
			}
		}
	}
}

 


 

  • 생각

1. dfs로 푸는 방법 - 정확하게 풀 수 있다. 벽을 부수고 지나왔으면 그 다음 벽은 못 부수게 하면되고 안부수고 지나왔으면 다음 벽을 만났을 때 부술 수 있게 돌리면 된다. 최종으로 도달했을 때 가장 짧은 길이를 출력 ==>> 시간 초과

 

위의 사진은 어떤 분이 정리 해 둔 것인데 보면서 깨달으면 좋을 것 같아서 가져왔다.

 

  1. 가중치가 없는 최단 경로는 무조건 BFS입니다. 왜 DFS가 안 될까요? 그 이유는 당연하게도, 특정 칸에 처음 도달했을 때까지의 경로의 길이가 다른 경로를 통해 도달한 길이보다 짧다는 보장이 전혀 없기 때문입니다. 아직까지 이 사실을 모르고 계셨다면, 이 문제는 아직 풀기에 너무 어렵습니다. 더 기본적인 BFS 문제을 먼저 풀어보세요.
  2. 모든 칸을 전부 0으로 하나씩 바꾸어보고 BFS를 돌리는 것을 반복해서는 통과될 수 없습니다. 대부분의 알고리즘 문제가 그렇듯이, 풀이를 짜기 전에 반드시 해야 하는 것 중 하나는 시간 복잡도를 생각하는 것입니다. 시간 복잡도 계산, 전혀 어렵지 않습니다. 벽이 최대 O(NM)개 있는 맵에서, 벽을 하나 부술 때마다 O(NM)개의 칸을 탐색해야 하죠? 그러니 O((NM)^2)입니다. 이 수는 우리가 대충 1초에 돌 수 있다고 보는 단위인 1억을 10000배나 뛰어넘는 1조입니다. 절대 통과될 수 없겠죠?
  3. 칸마다 방문 체크 하나씩만 하는 방법으로는 풀 수 없습니다. 어떤 칸에 도달했을 때 나는 "아직 벽을 부술 수 있는 상태"일 수도 있고, "더 이상 벽을 부술 수 없는 상태"일 수도 있습니다. 큐에 그 상태를 넣은 것만으로 되는 것이 아닙니다. 당장 이 지점까지 어떻게든 최단으로 오는 길만 구했다고 해서, 그 이후의 여정도 최적으로 만들 수 있는 건 아닙니다. 구체적인 예시가 필요한가요?
    1. 현재 칸까지 벽을 안 부수고 최단으로 올 수 있었다고 가정해봅시다. 현재 지점에서 목표 지점까지 가는 데에, 벽을 한 개 부수고 가는 것이 안 부수고 가는 것보다 최적이 나온다고 해봅시다. 그렇다면 지금 내가 벽을 더 부술 수 있는 상태라는 사실을 알고 있어야만 되겠죠?
    2. 벽을 안 부수고도 현재 칸까지 도달이 가능하지만, 벽을 부수고 오는 것이 더 짧다고 가정해봅시다. 현재 지점에서 목표 지점까지 가려면 무조건 벽을 한 개 부숴야만 된다고 해봅시다. 비록 현재 칸까지는 벽을 부수고 오는 것이 최적이었지만, 이 상태로는 끝에 아예 도달을 못 하죠? 현재 칸까지는 더 멀더라도 벽을 안 부수고 와야, 끝에 도달이 가능하죠.
  4. (스포일러) 그래서 이 문제에서는 BFS에 대해 새로운 테크닉을 요구합니다. 단순히 좌표만을 큐에 넣어 탐색하는 방식을 넘어, "현재 상태" 자체를 큐에 넣어서 문제를 풀어야 합니다. 즉, 어떤 좌표에 있는가 뿐만 아니라, "여기까지 오면서 벽을 부순 적이 있는가" 여부를 함께 큐에 저장해서 탐색하고, 각각을 별개로 방문 체크해줘야 하는 문제입니다. visited[x][y]가 아니라, visited[x][y][벽을 부순 적이 있는가?] 가 되어야 합니다.
  5. 이 문제에서는 같은 칸에 방문하는 경우 벽을 안 부순 것이 더 유리하기 때문에 벽을 부쉈는지 여부를 방문 배열에 기록하여 부순 횟수가 더 적을 때만 방문하는 방법도 됩니다. 그러나 이는 문제의 특성 때문에 이 문제에서만 통하는 그리디이므로 다른 문제에도 함부로 사용해서는 안 됩니다.

2. bfs로 푸는 방법 - 조건을 잘 설정하면 dfs보다 메모리는 더 쓰겠지만 속도는 빠를 것 같다. (시간과 메모리 제한 보고 알고리즘을 생각하는 정도의 수준까지 오면 좋을 것 같다. ==>> 공부를 열심히 해야할 듯.)

 

 

  • 코드

정답 코드 : bfs로 풀었다. 여기서 핵심은 중복제거를 check[N][M][2]로 벽을 지나오면서 왔는지 아닌지 중복체크를 해주는 방법을 사용해야지만 정확히 풀 수 있는 문제이다. 

 

import java.io.BufferedReader;
import java.io.InputStreamReader;
import java.util.LinkedList;
import java.util.Queue;
import java.util.StringTokenizer;

public class Main {
	static int N, M, minValue;
	static int[][] array;
	static boolean[][][] check;
	static int[] x = { -1, 1, 0, 0 };
	static int[] y = { 0, 0, -1, 1 };

	public static void main(String[] args) throws Exception {
		SetData();
		bfs();
        if(minValue == Integer.MAX_VALUE)
            System.out.println(-1);
        else    
            System.out.println(minValue);
	}

	private static void SetData() throws Exception {
		BufferedReader br = new BufferedReader(new InputStreamReader(System.in));
		StringTokenizer st = new StringTokenizer(br.readLine());

		N = Integer.parseInt(st.nextToken());
		M = Integer.parseInt(st.nextToken());
		array = new int[N][M];
		check = new boolean[N][M][2];
		minValue = Integer.MAX_VALUE;
		String s;
		
		for (int i = 0; i < N; i++) {
			s = br.readLine();
			for (int j = 0; j < M; j++) {
				array[i][j] = s.charAt(j) - '0';
			}
		}
	}

	private static void bfs() {
		Queue<int[]> queue = new LinkedList<>();
		queue.offer(new int[] { 0, 0, 1, 0 });

		while (!queue.isEmpty()) {
			int location[] = queue.poll();
			
			int count = location[2];
			int breakWall = location[3];
			if (location[0] == N - 1 && location[1] == M - 1) {
				minValue = Math.min(minValue, count);		// 마지막 위치에 도달했을 때 지나온 칸 수를 return
			}

			for (int direction = 0; direction < 4; direction++) {
				int r = location[0] + x[direction];
				int c = location[1] + y[direction];

				if (r < 0 || r >= N || c < 0 || c >= M)
					continue;
				if (array[r][c] == 1 && breakWall == 1)	// 지나온 벽이 있고 다음 칸도 벽이면 진행 X
					continue;

				if (breakWall == 1) {
					if (!check[r][c][breakWall] && array[r][c] == 0) {
						check[r][c][breakWall] = true;
						queue.offer(new int[] { r, c, count + 1, breakWall});
					}
				} else { // 벽을 안부순 상태
					if (!check[r][c][breakWall] && array[r][c] == 1) {
						check[r][c][breakWall] = true;
						queue.offer(new int[] { r, c, count + 1, breakWall + 1 });
					} else if (!check[r][c][breakWall] && array[r][c] == 0) {
						check[r][c][breakWall] = true;
						queue.offer(new int[] { r, c, count + 1, breakWall });
					}
				}
			}
		}
	}
}

 


 

  • 생각

1. 그냥 반복문 O(N^2)만큼 돌리면 시간 초과?? ==>> 시간 초과

 

2. 이분 탐색으로 하면?? 

 

3. Arraylist를 이용해서 max 값과 listsize를 빼가며 값을 정확히 도출하는 방법 ==>> 시간 초과

 

  • 코드

정답 코드 : 이분 탐색으로하면 시간초과 나지 않고 잘 동작한다.

 

import java.io.BufferedReader;
import java.io.InputStreamReader;
import java.util.StringTokenizer;

public class Main {
	static int N;
	static long count;
	static long[] number, temp;

	public static void main(String[] args) throws Exception {
		SetData();
		Sort(0, N-1);
		System.out.println(count);
	}

	private static void SetData() throws Exception {
		BufferedReader br = new BufferedReader(new InputStreamReader(System.in));
		StringTokenizer st = null;

		N = Integer.parseInt(br.readLine());
		number = new long[N];
		temp = new long[N];
		count = 0;

		st = new StringTokenizer(br.readLine());
		for (int i = 0; i < N; i++)
			number[i] = Long.parseLong(st.nextToken());
	}

	public static void Sort(int start, int end) {
		if(start < end) {
			int middle = (start+end)/2;
			Sort(start, middle);
			Sort(middle+1, end);
			merge(start,middle,end);
		}
	}

	private static void merge(int start, int middle, int end) {
		int startToMidStartIndex = start;
		int midToEndStartIndex = middle+1;
		int startTempIndex = start;
	
		while(startToMidStartIndex<=middle && midToEndStartIndex<=end) {
			if(number[startToMidStartIndex] <= number[midToEndStartIndex]) {
				temp[startTempIndex++] = number[startToMidStartIndex++];
			}else {
				temp[startTempIndex++] = number[midToEndStartIndex++];
				count +=  (middle + 1 - startToMidStartIndex);
			}
		}
		while(startToMidStartIndex<=middle)
			temp[startTempIndex++] = number[startToMidStartIndex++];
		while(midToEndStartIndex<=end)
			temp[startTempIndex++] = number[midToEndStartIndex++];
		
		for(int k=start; k<=end; k++) {
			number[k] = temp[k];
		}
		
	}
}

 

 


 

  • 생각

1. 문제를 일고 처음에 이해가 되지 않았는데, 예제를 보고선 도킹을 할 때 현재 번호보다 낮은 번호에만 도킹을 할 수 있다는 사실을 알게되었다.

 

  • 코드

정답 코드 : gate[index] == index 일때만 도킹을 해준다. 도킹을 했을댄 index의 값을 -1해줘서 해당 인덱스에는 다신 도킹할 수 없게 하였다. gate[index] != index 경우에는 해당 게이트보다 낮은 게이트를 재귀로 모두 다 돌아본 다음 도킹 가능한 게이트가 있으면 도킹, index가 0까지 가는 경우 도킹을 종료한다.

 

import java.io.BufferedReader;
import java.io.InputStreamReader;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Comparator;
import java.util.HashSet;
import java.util.LinkedList;
import java.util.PriorityQueue;
import java.util.Queue;
import java.util.StringTokenizer;

public class Main {
	static int G, P;
	static int[] gate, airplane;

	public static void main(String[] args) throws Exception {
		SetData();
		System.out.println(FindMaxValue());
	}

	private static void SetData() throws Exception {
		BufferedReader br = new BufferedReader(new InputStreamReader(System.in));
		G = Integer.parseInt(br.readLine());
		P = Integer.parseInt(br.readLine());
		gate = new int[G + 1];
		airplane = new int[P + 1];

		for (int i = 1; i <= G; i++)
			gate[i] = i;

		for (int i = 1; i <= P; i++)
			airplane[i] = Integer.parseInt(br.readLine());
	}

	private static int FindMaxValue() {
		int count = 0;

		for (int i = 1; i <= P; i++) {
			if (docking(airplane[i]) < 0)
				break;
			count++;
		}

		return count;
	}

	private static int docking(int index) {
		// basecase (게이트 번호의 값이 0이면 도킹 불가)
		if (gate[index] == 0)	return -1;
		
		if (gate[index] == index)	// 도킹을 아직 시도하지 않은 게이트라면
			gate[index]--;
		else						// 도킹을 한 게이트라면 현재 게이트보다 낮은 게이트를  찾기 위해 재귀를 돔.
			gate[index] = docking(gate[index]);
		
		return gate[index];
	}
}

 


 

  • 생각

1. DFS - 멀티탭의 모든 칸이 꽉 찰때마다 모든 멀티탭의 플러그를 한번씩 빼가며 DFS를 돌림. 정확하지만 시간초과 가능성 높음

 

2. 멀티탭에 꽂으면서 꽉찼을 때만 뽑을 콘센트를 찾는 작업을 진행함. 콘센트를 빼는 것은 뒤에 가장 적게오는 콘센트를 뽑는다.

 

 

  • 코드

정답 코드 : 2번 방법으로 콘센트가 꽉 찼을 때만 뒤에 남아있는 콘센트 번호 중에 가장 적게 나오는 번호를 뽑는다.

 

import java.io.BufferedReader;
import java.io.InputStreamReader;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Comparator;
import java.util.HashSet;
import java.util.LinkedList;
import java.util.PriorityQueue;
import java.util.Queue;
import java.util.StringTokenizer;

public class Main {
	static int N, K;
	static int[] array;
	static HashSet<Integer> powerSocket;	// 멀티탭에 먼저 꽂았는지 꽂지 않았는지는 상관이 없다. HashSet 사용

	public static void main(String[] args) throws Exception {
		SetData();
		System.out.println(FindMinValue());
	}

	private static void SetData() throws Exception {
		BufferedReader br = new BufferedReader(new InputStreamReader(System.in));
		StringTokenizer st = new StringTokenizer(br.readLine());
		N = Integer.parseInt(st.nextToken());
		K = Integer.parseInt(st.nextToken());
		powerSocket = new HashSet<Integer>();
		array = new int[K];

		st = new StringTokenizer(br.readLine());
		for (int i = 0; i < K; i++)
			array[i] = Integer.parseInt(st.nextToken());

	}

	private static int FindMinValue() {
		int count = 0;
		
		for(int i = 0; i < K; i++) {
			// 콘센트가 꽉차지 않거나 콘센트에 꽂혀있는 번호이면 continue
			if(powerSocket.contains(array[i])  || isPossible(array[i])) 
				continue;
			
			// 콘센트가 꽉차고 콘센트에 꽂혀있는 번호가 아닐 시
			int max = -1, pick = -1;
			for (int num : powerSocket) {
				int temp = 0;	// temp를 통해 뒤에 가장 적게 오는 수를 지워줌
				for (int j = i + 1; j < K; j++) {
					if (num == array[j]) {
						break;
					}
					temp++;
				}
				if (temp > max) {
					pick = num;
					max = temp;
				}

			}
			powerSocket.remove(pick);
			powerSocket.add(array[i]);

			count++;
		}
		
		return count;
	}
	
	private static boolean isPossible(int item) {
		if (powerSocket.size() < N) {
			powerSocket.add(item);
			return true;
		}
		return false;
	}
	
}

 

'algorithm' 카테고리의 다른 글

[JAVA] 백준 1517번 : 버블 소트  (0) 2020.11.12
[JAVA] 백준 10775번 : 공항  (0) 2020.11.10
[JAVA] 백준 1202번 : 보석 도둑  (0) 2020.11.09
[JAVA] 백준 15683번 : 감시  (0) 2020.11.06
[JAVA] 백준 2011번 : 암호코드  (0) 2020.11.05

 


 

  • 생각

1. 가방에 넣을 수 있는 보석을 차례대로 dfs로 넣는다? => 모든 경우의 수를 다 탐색한다. 정확하지만 시간초과의 가능성 존재

 

2. 가방의 무게가 수용할 수 있는 최소 무게 가방부터 넣을 수 있는 보석을 확인 후 넣을 수 있는 최대의 가격 보석을 넣는다. 

tip) 2차원 배열 정렬

 

  • 코드

정답 코드 : 1번 방법은 시간초과가 나올 것 같아서 2번 방법을 시행했다. 2번 방법을 쉽게 푸는 팁이 있다면 array.sort로 배열을 오름차순으로 정렬 후 pritorityqueue 우선순위 큐(큐에 있는 값들이 2개 이상이면 오름차순으로 큐에 들어감)를 사용하여 쉽게 해결할 수 있었다.

 

import java.io.BufferedReader;
import java.io.InputStreamReader;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Comparator;
import java.util.LinkedList;
import java.util.PriorityQueue;
import java.util.Queue;
import java.util.StringTokenizer;

public class Main {
	static int N, K;
	static int[][] jewelry;
	static int[] bag;
	static long maxValue;

	public static void main(String[] args) throws Exception {
		SetData();
		FindMaxValue();
		System.out.println(maxValue);
	}

	private static void SetData() throws Exception {
		BufferedReader br = new BufferedReader(new InputStreamReader(System.in));
		StringTokenizer st = new StringTokenizer(br.readLine());
		N = Integer.parseInt(st.nextToken());
		K = Integer.parseInt(st.nextToken());
		maxValue = 0;
		jewelry = new int[N][2];
		bag = new int[K];

		for (int i = 0; i < N; i++) {
			st = new StringTokenizer(br.readLine());
			jewelry[i][0] = Integer.parseInt(st.nextToken());
			jewelry[i][1] = Integer.parseInt(st.nextToken());
		}

		for (int i = 0; i < K; i++)
			bag[i] = Integer.parseInt(br.readLine());

		// 2차원 배열 sort
		Arrays.sort(jewelry, new Comparator<int[]>() {
			@Override
			public int compare(int[] o1, int[] o2) {
				// TODO Auto-generated method stub
				return o1[0] - o2[0];
			};

		});
		Arrays.sort(bag);
	}

	private static void FindMaxValue() {
		// 우선 순위 큐로 해야지 오름차순 정렬이 되서 작은 값이 맨 먼저 나오게됌
		Queue<Integer> queue = new PriorityQueue<>();
		int j = 0;
		for (int i = 0; i < K; i++) {
			while (j < N && jewelry[j][0] <= bag[i]) { // 앞에서 담은것은 제외해야 하므로 while문과 j를 사용
				queue.add(-jewelry[j][1]);	// -를 해주는 이유는 -를 해야지 양수 최대값이 음수로 넣었을 때 오름차순에서 맨 앞으로 가져올 수 있게한다. 
				j++;
			}
			if (!queue.isEmpty()) { // for문 한번에 한번만 더한다.
				maxValue += Math.abs(queue.poll());	// 음수로 바꿔줬던 수에 절대값을 씌워서 양수로 다시 나오게함
			}
		}

	}
}

 

'algorithm' 카테고리의 다른 글

[JAVA] 백준 10775번 : 공항  (0) 2020.11.10
[JAVA] 백준 1700번 : 멀티탭 스케줄링  (0) 2020.11.09
[JAVA] 백준 15683번 : 감시  (0) 2020.11.06
[JAVA] 백준 2011번 : 암호코드  (0) 2020.11.05
[JAVA] 백준 1062번 : 가르침  (0) 2020.11.04

 


 

  • 생각

1. DFS로 돌리는 경우 - 1번 4개, 2번 2개, 3번 4개, 4번 4개, 5번 1개로 모든 방향으로 확인하면서 복구 시킨다. 

 ==>> 시간초과 뜰 것 같았는데 역시 시간초과가 뜬다. 어떻게 고쳐야하지?

 

  • 코드

정답 코드 : 똑같이 dfs로 풀고 비슷한 방식으로 풀었는데 이건 왜 빠른지 찾아보자.. 아직 이해안함

 

import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
import java.util.StringTokenizer;

public class Main {
	static int R, C, result, cnt, tmpResult;
	static int[][] grid;
	static int[][][] camPos;
	static int[] camCount, camAcc;
	static int[][] camDir;
	static int[] dirTypes = {4, 2, 4, 4, 1};
	public static void main(String[] args) throws IOException {
		BufferedReader br = new BufferedReader(new InputStreamReader(System.in));
		StringTokenizer st = new StringTokenizer(br.readLine().trim());
		
		R = Integer.parseInt(st.nextToken());
		C = Integer.parseInt(st.nextToken());
		
		grid = new int[R][C];
		
		result = Integer.MAX_VALUE;
		
		camPos = new int[5][8][2];
		camDir = new int[5][8];
		
		for (int i = 0; i < 5; i++) {
			for (int j = 0; j < 8; j++) {
				camDir[i][j] = -1;
			}
		}
		
		camCount = new int[5];
		
		for (int r = 0; r < R; r++) {
			st = new StringTokenizer(br.readLine().trim());
			for (int c = 0; c < C; c++) {
				grid[r][c] = Integer.parseInt(st.nextToken());
				
				if(grid[r][c] == 0)
					continue;
				else if(grid[r][c] == 6)
					continue;
				else {
					camPos[grid[r][c]-1][camCount[grid[r][c]-1]][0] = r;
					camPos[grid[r][c]-1][camCount[grid[r][c]-1]][1] = c;
//					System.out.println(r+","+c);
					camCount[grid[r][c]-1] ++;
				}
			}
		}
		
		camAcc = new int[5];
		
		for (int i = 0; i < 5; i++) {
			camAcc[i] = camCount[i];
		}
		
		for (int i = 1; i < 5; i++) {
			camAcc[i] = camAcc[i-1] + camAcc[i];
		}
		
		DFS(0);
		
		System.out.println(result);
	}
	private static void DFS(int count) {
		if(count == camAcc[4]) {
			
//			for (int i = 0; i < 5; i++) {
//				for (int j = 0; j < 8; j++) {
//					System.out.print(camDir[i][j]+" ");
//				}
//				System.out.println();
//			}
//			System.out.println();
			
			int[][] tmpGrid = new int[R][C];
			for (int r = 0; r < R; r++) {
				for (int c = 0; c < C; c++) {
					tmpGrid[r][c] = grid[r][c];
				}
			}
			
			tmpResult = 0;
			
			for (int i = 0; i < 5; i++) {
				for (int j = 0; j < camCount[i]; j++) {
					if(i == 0) {
						dirCheck(camDir[i][j], tmpGrid, camPos[i][j][0], camPos[i][j][1]);
					} else if(i == 1) {
						dirCheck(camDir[i][j], tmpGrid, camPos[i][j][0], camPos[i][j][1]);
						dirCheck(camDir[i][j]+2, tmpGrid, camPos[i][j][0], camPos[i][j][1]);
					} else if(i == 2) {
						dirCheck(camDir[i][j], tmpGrid, camPos[i][j][0], camPos[i][j][1]);
						dirCheck((camDir[i][j]+1)%4, tmpGrid, camPos[i][j][0], camPos[i][j][1]);
					} else if(i == 3) {
						dirCheck(camDir[i][j], tmpGrid, camPos[i][j][0], camPos[i][j][1]);
						dirCheck((camDir[i][j]+1)%4, tmpGrid, camPos[i][j][0], camPos[i][j][1]);
						dirCheck((camDir[i][j]+2)%4, tmpGrid, camPos[i][j][0], camPos[i][j][1]);
					} else {
						dirCheck(camDir[i][j], tmpGrid, camPos[i][j][0], camPos[i][j][1]);
						dirCheck((camDir[i][j]+1)%4, tmpGrid, camPos[i][j][0], camPos[i][j][1]);
						dirCheck((camDir[i][j]+2)%4, tmpGrid, camPos[i][j][0], camPos[i][j][1]);
						dirCheck((camDir[i][j]+3)%4, tmpGrid, camPos[i][j][0], camPos[i][j][1]);
					}
				}
			}
			
			for (int r = 0; r < R; r++) {
				for (int c = 0; c < C; c++) {
					if(tmpGrid[r][c] == 0)
						tmpResult++;
				}
			}
			
			if(result > tmpResult) {
				result = tmpResult;
//				for (int r = 0; r < R; r++) {
//					for (int c = 0; c < C; c++) {
//						System.out.print(tmpGrid[r][c]+" ");
//					}
//					System.out.println();
//				}
//				
//				System.out.println();
//				
			}
			
			return;
		}
		int type = -1;
		int idx = -1;
		
//		System.out.println(count);
		
		for (int i = 0; i < 5; i++) {
			if(count < camAcc[i]) {
				type = i;
				if(i == 0)
					idx = count;
				else
					idx = count - camAcc[i-1];
				break;
			}
		}
		
		for (int i = 0; i < dirTypes[type]; i++) {
			camDir[type][idx] = i;
			DFS(count + 1);
		}
		
	}
	private static void dirCheck(int dir, int[][] tmpGrid, int currR, int currC) {
		int[] dr = {-1, 0, 1, 0};
		int[] dc = {0, 1, 0, -1};
		
		int tr = currR;
		int tc = currC;
		
		while(true) {
			tr += dr[dir];
			tc += dc[dir];
			
			if(tr >= R || tc >= C || tr < 0 || tc < 0)
				break;
			
			int currVal = tmpGrid[tr][tc];
			
			if(currVal == 6)
				break;
			
			if(currVal == 0) {
				tmpGrid[tr][tc] = 9;
			}
			else 
				continue;
		}
	}
}

 

 

 

실패 코드 : 앞선 1의 시간초과뜬 코드이다. 시간을 줄일 방법이 생각이 안난다..

 

import java.io.BufferedReader;
import java.io.InputStreamReader;
import java.util.StringTokenizer;

public class Main {
	static int N, M, count, minValue;
	static int[][] array;
	static int[] x, y;

	public static void main(String[] args) throws Exception {
		SetData();
		DFS(0);
		System.out.println(minValue);
	}

	// 데이터
	private static void SetData() throws Exception {
		BufferedReader br = new BufferedReader(new InputStreamReader(System.in));
		StringTokenizer st = new StringTokenizer(br.readLine());

		N = Integer.parseInt(st.nextToken());
		M = Integer.parseInt(st.nextToken());
		count = 0; // 벽이 아닌 방향이 있는 1~5 숫자가 몇개있는지 count (basecase로 두기 위함)
		array = new int[N][M];
		x = new int[8];
		y = new int[8];
		minValue = Integer.MAX_VALUE; // 사각지대 최소 개수

		for (int i = 0; i < N; i++) {
			st = new StringTokenizer(br.readLine());
			for (int j = 0; j < M; j++) {
				array[i][j] = Integer.parseInt(st.nextToken());

				// 미리 cctv 좌표를 저장한다.
				if (array[i][j] != 0 && array[i][j] != 6) {
					x[count] = i;
					y[count] = j;
					count++;
				}
			}
		}
	}

	private static void DFS(int startIndex) {

		// basecase 모든 cctv를 확인해본 결과
		if (startIndex == count) {
			minValue = Math.min(minValue, RemainCCTV());
			return;
		}

		for (int i = startIndex; i < count; i++) {
			if (array[x[i]][y[i]] == 1) {
				CheckCCTV(x[i], y[i], startIndex, 1);
			} else if (array[x[i]][y[i]] == 2) {
				CheckCCTV(x[i], y[i], startIndex, 2);
			} else if (array[x[i]][y[i]] == 3) {
				CheckCCTV(x[i], y[i], startIndex, 3);
			} else if (array[x[i]][y[i]] == 4) {
				CheckCCTV(x[i], y[i], startIndex, 4);
			} else if (array[x[i]][y[i]] == 5) {
				CheckCCTV(x[i], y[i], startIndex, 5);
			}
		}
	}

	private static void CheckCCTV(int i, int j, int startIndex, int numberOfCCTV) {
		switch (numberOfCCTV) {
		case 1:
			GoUp(i,j);
			DFS(startIndex + 1);
			GoBackUp(i,j);
			GoDown(i,j);
			DFS(startIndex + 1);
			GoBackDown(i,j);
			GoLeft(i,j);
			DFS(startIndex + 1);
			GoBackLeft(i,j);
			GoRight(i,j);
			DFS(startIndex + 1);
			GoBackRight(i,j);
			break;
		case 2:
			GoUp(i,j);
			GoDown(i,j);
			DFS(startIndex + 1);
			GoBackUp(i,j);
			GoBackDown(i,j);
			GoLeft(i,j);
			GoRight(i,j);
			DFS(startIndex + 1);
			GoBackLeft(i,j);
			GoBackRight(i,j);
			break;
		case 3:
			GoDown(i,j);
			GoLeft(i,j);
			DFS(startIndex + 1);
			GoBackDown(i,j);
			GoBackLeft(i,j);
			GoUp(i,j);
			GoRight(i,j);
			DFS(startIndex + 1);
			GoBackUp(i,j);
			GoBackRight(i,j);
			GoLeft(i,j);
			GoUp(i,j);
			DFS(startIndex + 1);
			GoBackUp(i,j);
			GoBackLeft(i,j);
			GoRight(i,j);
			GoDown(i,j);
			DFS(startIndex + 1);
			GoBackRight(i,j);
			GoBackDown(i,j);
			break;
		case 4:
			GoUp(i,j);
			GoDown(i,j);
			GoLeft(i,j);
			DFS(startIndex + 1);
			GoBackUp(i,j);
			GoBackDown(i,j);
			GoBackLeft(i,j);
			GoUp(i,j);
			GoDown(i,j);
			GoRight(i,j);
			DFS(startIndex + 1);
			GoBackUp(i,j);
			GoBackDown(i,j);
			GoBackRight(i,j);
			GoLeft(i,j);
			GoRight(i,j);
			GoUp(i,j);
			DFS(startIndex + 1);
			GoBackUp(i,j);
			GoBackLeft(i,j);
			GoBackRight(i,j);
			GoLeft(i,j);
			GoRight(i,j);
			GoDown(i,j);
			DFS(startIndex + 1);
			GoBackLeft(i,j);
			GoBackRight(i,j);
			GoBackDown(i,j);
			break;
		case 5:
			GoUp(i,j);
			GoDown(i,j);
			GoLeft(i,j);
			GoRight(i,j);
			DFS(startIndex + 1);
			GoBackUp(i,j);
			GoBackDown(i,j);
			GoBackLeft(i,j);
			GoBackRight(i,j);
			break;
		default:
			break;
		}
	}

	private static void GoUp(int startI, int startJ) {
		for (int i = startI; i >= 0; i--) {
			if (array[i][startJ] == 6)
				break;
			if (array[i][startJ] == 0)
				array[i][startJ] = 7;
		}
	}

	private static void GoDown(int startI, int startJ) {
		for (int i = startI; i > N; i++) {
			if (array[i][startJ] == 6)
				break;
			if (array[i][startJ] == 0)
				array[i][startJ] = 7;
		}
	}

	private static void GoLeft(int startI, int startJ) {
		for (int j = startJ; j >= 0; j--) {
			if (array[startI][j] == 6)
				break;
			if (array[startI][j] == 0)
				array[startI][j] = 7;
		}
	}

	private static void GoRight(int startI, int startJ) {
		for (int j = startJ; j < M; j++) {
			if (array[startI][j] == 6)
				break;
			if (array[startI][j] == 0)
				array[startI][j] = 7;
		}
	}

	private static void GoBackUp(int startI, int startJ) {
		for (int i = startI; i >= 0; i--) {
			if (array[i][startJ] == 6)
				break;
			if (array[i][startJ] == 7)
				array[i][startJ] = 0;
		}
	}

	private static void GoBackDown(int startI, int startJ) {
		for (int i = startI; i > N; i++) {
			if (array[i][startJ] == 6)
				break;
			if (array[i][startJ] == 7)
				array[i][startJ] = 0;
		}
	}

	private static void GoBackLeft(int startI, int startJ) {
		for (int j = startJ; j >= 0; j--) {
			if (array[startI][j] == 6)
				break;
			if (array[startI][j] == 7)
				array[startI][j] = 0;
		}
	}

	private static void GoBackRight(int startI, int startJ) {
		for (int j = startJ; j < M; j++) {
			if (array[startI][j] == 6)
				break;
			if (array[startI][j] == 7)
				array[startI][j] = 0;
		}
	}

	private static int RemainCCTV() {
		int countOfCCTV = 0;

		for (int i = 0; i < N; i++) {
			for (int j = 0; j < M; j++) {
				if (array[i][j] == 0)
					countOfCCTV++;
			}
		}

		return countOfCCTV;
	}

}

+ Recent posts