- https://www.acmicpc.net/problem/11720
- BufferedReader 사용법은 자주 사용하지 않으니 매번 헷갈린다. 메모해 두고 복습하자.
BufferedReader br = new BufferedReader(new InputStreamReader(System.in));
- char 타입을 int 타입으로 변환하는 방법은 두 가지가 있다.
// 1. Ascii code를 이용하는 방법
char c1 = '9';
int num = c1 - '0';
// 2. Character.getNumericValue()를 이용하여 처리하는 방법
char c1 = '9';
int num = Character.getNumericValue(c1);
- https://www.acmicpc.net/problem/1546
- 주어진 공식대로 배열을 순회하면서 max 값보다 작으면 점수를 조작했는데, 식을 간소화 하면 이런 불필요한 로직을 줄일 수 있었다.
(A / M * 100 + B / M * 100 + C / M * 100) / 3 = (A + B + C) * 100 / M /3
- https://www.acmicpc.net/problem/11659
- StringTokenizer 사용 시 불필요한 hasMoreTokens() 체크 로직은 빼자.
- 합배열 공식 :
S[i] = S[i - 1] + A[i]
- 구간 합 공식 :
S[toIndex] - S[fromIndex - 1]
- https://www.acmicpc.net/problem/11660
- 처음 구간합을 구할 때 행의 합만 구하고 열의 합을 구할 아이디어를 떠올리지 못했다.
- 2차원 배열 index :
int[][] arr = new int[row][column]
- 합배열 공식 :
D[i][j] = D[i][j-1] + D[i-1][j] - D[i-1][j-1] + A[i][j]
- X1, Y1, X2, Y2 구간 합을 구하는 공식 :
D[X2][Y2] - D[X1-1][y2] - D[X2][Y1-1] + D[X1-1][Y1-1]
- https://www.acmicpc.net/problem/10986
- 처음에는 구간합 배열에 구간합을 구해서 저장을 했는데 굳이 그럴 필요가 없었다. 문제가 요구하는 것은 결국 %M 했을 때의 나머지가 0인(나누어 떨어지는 수의) 개수를 구하는 것이다. 따라서 나머지 배열에 구간합 % M을 한 나머지를 인덱스로 값을 증가시키면 되었다.
- 임의의 수 x1, x2 를 입력 받았을 때
(x1 + x2) % M 은 ((x1 % M) + (x2 % M)) % M
과 같다. 이게 핵심 아이디어였다. - 처음에는 if를 사용하여 나머지가 0일때 정답 변수의 값을 1 증가 시켰는데 굳이 그럴 필요가 없었다. 구간합을 구하면서 나누어 떨어진 수가 0이라면 나머지 저장 배열의 0번 인덱스에 저장될 것이므로, 계산 루틴이 끝난 후 0번 인덱스의 값을 꺼내고 + 각 인덱스에 저장된 값의 경우의 수를 계산하여 더해주면 되는 것이었다.
- https://www.acmicpc.net/problem/2018
- 투 포인터를 이용하여 O(n)의 시간복잡도로 특정 구간합을 계산할 수 있다는 것을 배웠다.
- 단, 투 포인터를 이용하려면 배열이 오름차순 정렬되어 있어야 한다.
- https://www.acmicpc.net/problem/1940
- 문제를 제대로 읽지 않아 두 수의 합을 구하는 것을 지나쳐서 어떤 알고리즘을 적용해야 하는지 고민만 했다. 문제의 요구사항과 예제 입출력을 잘 살펴보자.
- https://www.acmicpc.net/problem/1253
- 투 포인터의 위치와 좋은 수의 배열에서의 위치는 별개로 생각해야 하는데 투 포인터의 위치를 좋은 수의 위치와 합쳐서 생각하는 오류가 있었다.
- 최초 부분 문자열 처리 loop가 종료된 후 ACGT의 문자열 개수가 요구하는 문자열 개수와 맞는지 체크해야 했는데, loop 도는 중에 계속 ACGT 문자의 개수를 체크하는 실수를 했다.
- 디버거 돌렸을 때도 체크된 값은 정상적으로 나왔기에 로직 이상을 발견하는데 시간이 많이 걸렸다.
- https://www.acmicpc.net/problem/11003
- ArrayDeque 클래스는 스택으로 사용할 경우 Stack 보다 빠르며, Queue 로 사용 할 경우 LinkedList 보다 빠를 가능성이 높다.
- ArrayDeque 에 Null 은 허용되지 않는다.
- https://www.baeldung.com/java-array-deque
- https://docs.oracle.com/javase/8/docs/api/java/util/ArrayDeque.html
- https://www.acmicpc.net/problem/1874
- 동시성 문제가 없다면 Stack 보다는 속도가 빠른 ArrayDeque 를 사용하자. Deque 의 addLast() 와 pollLast() 로 스택과 같은 기능을 구현할 수 있다.
- 코드 작성보다 문제의 지문과 출력 예제를 보고 문제의 요구 사항을 파악하는 게 더 어려웠다.
- https://www.acmicpc.net/problem/17298
- 반복문에서 출력을 해야 하는 경우 BufferedWriter를 쓰기 보다는 StringBuilder를 사용하여
append()
를 해주고 나중에System.out.print()
를 해주는 게 조금 더 빠른 것 같다. - 배열을 역순으로 탐색하면서 오큰수를 계산하는 아이디어를 떠올리지 못했다.
- https://www.acmicpc.net/problem/2164
- Queue와 Stack의
poll()
했을 때 데이터가 빠지는 위치를 헷갈려서 슈도 코드 작성에 애를 먹었다. Stack의 LIFO, Queue의 FIFO 특징을 잘 기억해두자.
- https://www.acmicpc.net/problem/11286
PriorityQueue
대해서 알고 있고, customizing 할 수 있는지가 문제 풀이의 핵심이었는데 이 아이디어를 떠올리지 못했다.
- https://www.acmicpc.net/problem/1377
- 처음에는 배열을 정렬해서 index 위치를 비교한다는 아이디어를 떠올리지 못해다.
- 굳이 배열을 정렬할 필요 없이 우선순위 큐를 이용하면 배열의 정렬에 드는 O(nlogn)의 시간을 줄일 수 있다.
- https://www.acmicpc.net/problem/2744
- 처음에 입력된 문자열을 char[] 배열로 변환했는데 굳이 변환할 필요가 없었다.
- Character.isLowerCase() / Character.isUpperCase()로 비교가 가능하다. 예전에 Character 클래스에 대해서 공부했는데 메서드를 모두 살펴보지 않아서 불필요한 로직을 구현하였다.
- https://www.acmicpc.net/problem/1919
- 최초 아이디어는 두 문자열을 비교해서 같은 문자가 있는지 없는지 비교하는 것이었는데 이 아이디어는 두 문자열의 길이가 같을 때만 유효한 아이디어였다.
- 두 문자열의 길이가 다를 수 있으므로 각 문자열의 알파벳 갯수를 세서 차이만큼 result에 더해주면 되었는데 이 아이디어를 떠올리지 못했다.
- https://www.acmicpc.net/problem/1543
String.equals()
로 비교할 생각만 하고indexOf
로 문자열의 위치를 검색하는 아이디어를 떠올리지 못했다.replace()
함수로 문자열을 지우고 남은 문자열의 길이로 답을 구할 수도 있다.
- https://www.acmicpc.net/problem/1157
- 반복문을 돌 때 char를 직접 지정해서 돌면 많이 사용된 알파벳 저장시 명시적 type casting을 생략할 수 있다.
for (char alp = 'A'; alp <= 'Z'; alp++) { }
- https://www.acmicpc.net/problem/1236
- 각 행과 열에 필요한 경비원 수를 구하고 그 중 큰값을 출력해 주면 되었는데 이 아이디어를 떠올리지 못했다.
- https://www.acmicpc.net/problem/10431
- 삽입 정렬의 개념을 적용하면 풀 수 있는 문제였는데 삽입 정렬의 아이디어를 떠올리지 못했다. 삽입 정렬 알고리즘에 대해서 다시 복습하자.
- https://www.acmicpc.net/problem/10989
- 문자열 index를 사용해서 문자의 갯수를 셌던 아이디어를 수열에도 똑같이 적용할 수 있다. 일반적인 정렬 알고리즘의 최악의 시간복잡도는
$O(N^2)$ 이므로 입력되는 수의 개수가 일천만이 넘아간다면 일반적인 정렬 방법으로는 시간 초과가 날 가능성이 높다.
- https://www.acmicpc.net/problem/3273
- 시간 제한이 1초고 N의 크기가 백만이라 배열을 정렬하면 무조건 시간 초과가 난다. 백준 10989의 문제처럼 백열의 index를 활용해서 풀 수 있다.
pairValue = X - num[i]
라고 할 때exist[pairValue]
배열에 존재한다면 입력받은 모든 값들을 순회하지 않아도 답을 구할 수 있다.
- https://www.acmicpc.net/problem/2309
- 모두 더해서 100이 되는 수를 찾을 생각만 하고, 두 개의 수를 빼서 100이 되는 경우의 수를 찾는다는 아이디어를 떠올리지 못했다.
- 찾은 두 수를 -1이나 0으로 변경하고 sort를 하면 합해서 100이 되는 수는 무조건 0번과 1번 index에 위치할 것이므로 if 로직을 생략할 수 있다.
- https://www.acmicpc.net/problem/10448
- 문제를 잘못 이해해서 삼각함수로 표현될 수 있는 숫자로 판단을 했는데, 문제가 요구하는 사항은 삼각함수 3개의 합으로 표현될 수 있는 수인지 나타내는 것이었다.
- 시간제한 때문에 3중 for문을 쓸 생각을 하지 않았는데, 문제를 잘못 이해해서 N의 범위를 넓게 잡았기 때문이었다. 문제를 잘 읽고 생각을 깊게 하는 연습을 계속하자.
- https://www.acmicpc.net/problem/11005
- 진법변환 과정에서 나머지 연산결과를 저장하기 위해 stack 자료구조를 사용했는데
StringBuilder
를 이용해서 문자열로 바로 저장하고sb.reverse()
로 역순으로 출력하는 방법도 있었다.
- https://www.acmicpc.net/problem/11068
- 진법 변환시 전체 자리수가 몇 개인지 구할 때 N % B(진법) 연산을 반복하면서 자리수를 카운트 해주면 되었다.
- 회문인 검사할 때 center, left, right 인덱스를 모두 변수를 선언했는데 진법 변환된 수가 담긴 배열의 길이 / 2까지 for를 돌리고 이걸 left 인덱스로 쓰면서, 대칭인 right 인덱스는 배열의 길이 - for 변수 -1을 해주면 되었다.
- https://www.acmicpc.net/problem/3085
- 문제의 요구 사항을 이해하는데 시간이 많이 걸렸다. 구현 자체는 모든 경우를 다 탐색하면 되는데, 문제 이해를 잘못해서 잘못된 구현이 나오고 수정하는 일이 자주 발생하고 있다. 문제를 이해하는 훈련에 시간을 조금 더 투자할 필요가 있다.
- https://www.acmicpc.net/problem/10250
- 층수와 방번호를 구할 때 경계값을 생각하지 못하고 계산식을 도출해서 계속 오류가 발생했다. 예를 들어 W가 12, H가 10이라고 가정했을 때 110번째 손님의 방 번호는 1011호가 되어야 하지만 층수를 구하기 위해 단순히
110 % 10
을 하게 되면 몫이 0이 되어 1층에 손님이 배치되는 오류가 생긴다. 이런 오류를 방지라혀면(N - 1)
을 먼저 해주고 그 결과에%H
모듈러 연산을 한 다음+1
을 해주어야 한다. - N번째 손님의 층수: H로 나눈 나머지가 0이 나오는 케이스가 있으므로 [0, H-1] -> [1, H]가 나오도록 변환해야 한다.
int floor = (N - 1) % H + 1;
- N번째 손님의 호수 : H로 나눈 몫. 테스트 케이스가
1 ~ 10
이라고 했을 때1 ~ 9
까지는 몫이 0이 나오고 10만 몫이 1이 나온다. 층수를 구할 때와 마찬가지로N-1
을 먼저 해준 값을 H로 나누고 그 결과에 1을 더해준다. - 결론적으로
1 ~ H => 1 / H+1 ~ 2H => 2
로 변환하는 작업을 해줘야 한다. int distance = (N - 1) / H + 1;
- 문제에 나오는 테스트 케이스를 통과했다고 만족하지 말고 다양한 테스트 케이스를 직접 만들어서 테스트 하는 습관을 만들자.
- https://www.acmicpc.net/problem/1730
- 주어진 N의 크기를 벗어나는 좌표를 계산할 때 N의 크기는 1 ~ N 까지이지만, 배열의 index는 0부터 N-1 까지 계산해야 하는데, N으로 계산해서 오류가 계속 발생했다. 다음에는 조금 더 주의를 기울이자.
- https://www.acmicpc.net/problem/2840
- 환형 구조의 배열 index를 구할 때 modular 연산을 하면 되는데 원리를 몰라서 불필요한 계산 로직으로 구현을 했다.
- 바퀴의 방향과 화살표의 방향은 반대이므로
curIndex - S
를 해주는데 몇 바퀴나 돌았는지 모르니까(curIndex - S) % N
연산을 해준다. - 이 때 주의할 점은 Java의 modular 연산은
(curIndex - S)
의 값이 음수면 읍수 값을 리턴할 수 있기 때문에 음수 방지를 위해서 N을 더해서 양수로 만든 후 다시 % N으로 나누어 준다. - 환형 구조를 배열로 구현할 때 index를 구하는 공식은
currentIndex + x % N
. 잘 기억해 두자.
- https://www.acmicpc.net/problem/2817
- 문제가 요구하는 것은 각 스태프별로 득표수를
1~14
까지의 숫자로 나눈 값을 각각 배열로 저장한 후,1~14
위에 해당하는 값을 찾아 그 값에 해당하는 스태프의 칩을 1증가 시키는 것이었다. - 칩 분배 로직을 이해하지 못해서 구현에 애를 먹었는데, 한번만이라도 표를 손으로 그려보았다면 조금은 빨리 이해할 수 있는 문제였다. 다음에는 먼저 손으로 표를 그려보자.
- https://www.acmicpc.net/problem/2745
- char type 을 Ascii Code가 아닌 숫자로 변환하려면 두 가지 방법이 있다.
int a = Character.getNumericValue(x);
int a = x - '0';
- https://www.acmicpc.net/problem/1233
- 세 주사위의 합을 배열의 index로 사용해 놓고, 정작 index와 index에 저장된 값을 혼동해서 버그를 찾는데 불필요한 시간을 사용했다. 다음에는 슈도 코드를 꼭 작성해서 흐름을 머리속에 그려두고 코딩을 하자.
- https://www.acmicpc.net/problem/2231
- 어떤 수 x가 있다면
x % 10
연산을x / 10
연산의 결과가 0이 될 때까지 계속하면 각 자리수들의 합을 쉽게 구할 수 있었는데 이 아이디어를 떠올리지 못하고 각 자리수마다 0에서 9까지 반복하면서 재귀를 돌린다는 아이디어에 매몰되었다. 문제를 여러 관점에서 생각하는 힘을 기르자. - 어려운 문제가 아니었는데 로직의 도출 관점이 문제가 요구하는 방향과 어긋나면 쉬운 문제도 풀기 어렵다는 것을 배웠다.
- https://www.acmicpc.net/problem/1110
- while 보다는 do - while이 코드가 더 깔끔해지는 문제였다. 종료 조건이 처음에 N과 같고 계속 변화하다가 다시 N과 같아진다면 do - while을 고려해 보자.
- https://www.acmicpc.net/problem/4673
- 인자로 넘긴 수 x와, x의 각 자리수를 더하는 함수에서 넘겨받은 x에 합산을 하지 않고 0부터 합산하는 로직 오류가 있었다.
- https://www.acmicpc.net/problem/1018
- 실버 IV 문제인데 최초 아이디어를 떠올리고도 막상 구현을 하지 못했다. %2를 하면 row를 바꿔가면서 비교할 수 있다는 기초 지식이 있음에도 그걸 응용할 생각을 하지 못했다.
- https://www.acmicpc.net/problem/1120
- 배열의 index는 0부터 다룰지, 1부터 다룰지 확실하게 정하고 코드를 작성하자.
- https://www.acmicpc.net/problem/1181
- 중복 제거를 위해서 Set을 이용해서 입력을 받고 -> Set을 List로 변환 후 정렬했는데 변환 시간이 생각보다 오래 걸렸다.
- 배열에 모두 입력을 받고 배열의 정렬을 먼저 하면 같은 문자열끼리 모여 있기 때문에 중복을 제거하는데 오히려 효과적이었다.
- HashSet 보다 LinkedHashSet에서 중복을 제거하는 시간이 조금 더 걸렸다. 참고하자.
- https://www.acmicpc.net/problem/10814
Arrays.sort()
는 배열을,Collections.sort()
는 Collection을 정렬한다. TimSort는 stable한 특성을 갖고 있다.
- https://www.acmicpc.net/problem/7785
- Comparator와 Comparable의 사용법을 잘 복습해두자.
- https://www.acmicpc.net/problem/1302
- Comparator 정의 방법을 잘 기억해 두고 자주 연습하자.
Comparator<Map.Entry<String, Integer>> valueThenKeyComparator = (o1, o2) -> {
if (o2.getValue().equals(o1.getValue())) {
return o1.getKey().compareTo(o2.getKey());
}
return o2.getValue().compareTo(o1.getValue());
};
- Map의 메서드 중
getOrDefault()
는 아래의 로직을 깔끔하게 한줄로 줄여준다. Collection에서 지원하는 메서드가 어떤 게 있는지 조금 더 관심을 가지자.
if (books.containsKey(key)) {
int currentValue = books.get(key);
books.put(key, currentValue + 1);
} else {
books.put(key, 1);
}
- https://www.acmicpc.net/problem/18870
- 처음에 Map을 이용하여 구현했지만 시간 초과를 받았다. 이유는 입력된 값을 key로, 압축된 좌표를 value로 저장했다면
get()
메서드로 손쉽게 압축된 좌표를 꺼내올 수 있었는데 좌표를 key로, 입력된 값을 value로 저장하여 Map의 특성을 전혀 살리지 못하는 로직이 되었기 때문이다. - Map의 처음부터 N의 크기만큼 순회하니 결국 시간복잡도가
$O(N^2)$ 이 되어 시간 초과가 되었던 것이다. Java Collection에 대해서 좀 더 깊이 있는 공부가 필요하다는 것을 배웠다.
- https://www.acmicpc.net/problem/1931
- int 타입의 배열 정렬이 불완전 정렬이긴 하지만 속도는 Boxed 타입에 비해서 확실히 빠르다.
- https://www.acmicpc.net/problem/1431
- char type을 int 값으로 변환하는 방법은 2가지.
Charcter.getIntValue(x)
,charValue - '0'
. 자주 잊어버리는데 확실히 기억해 두자.
- https://www.acmicpc.net/problem/18310
- 문제에서 요구하는 것은 배열의 가장 중앙값을 구하는 것이었다. 기준위치를 인덱스로 계산해보면 인덱스의 좌측은 인데스 값이 +로, 우측은 -로 계산이 된다는 것을 알 수 있다.
- 반복되는 패턴이 있을 때, 그 패턴을 빨리 캐치하는 연습을 계속 하자.
- https://www.acmicpc.net/problem/11660
- 실제 변화량을 바로 기록하는 것이 아니라 실제 변화가 일어나야 하는 지점에 변화량을 기록한 뒤에, 그 각각의 변화량을 각 칸에 반영해 주는 형식을 누적 연산을 통해서 해결할 수 있다.
- 단, 항상 지시가 끝난 후에 매번 출력하는 경우에는 이 방법이 통할 수 없고, 모든 지시가 적용된 후에 결과만 구할 때에는 이렇게 누적합 개념을 응용을 해서 범위 연산 처리를 해볼 수 있다.
- 구간 합에 대한 쿼리만이 아니라 이런 형태로 구간 범위에 대한 변화량에도 응용을 해볼 수 있다.
- https://www.acmicpc.net/problem/1920
- Set를 이용하는 방법만이 아니라 Binary Search를 이용하여 원하는 수를 찾는 방법을 배웠다. binary search 는 알고리즘 문제에서 자주 등장하니 잘 학습해 두자.
Integer.valueOf()
보다Integer.parseInt()
가 근소하게 속도가 빠르지만 null safe 하지는 않다는 단점이 있다.
- https://www.acmicpc.net/problem/14425
- loop 내에서 어떤 연산의 결과를 반복적으로 사용해야 한다면 변수를 이용하자. '자바 성능 튜닝이야기'에도 나오는 내용이다.
- 이진 탐색에서 문자열을 비교한 결과를 변수에 할당하고 if 문에서 재사용했을 때 미세하더라도 속도의 향상이 있었다.
int compareResult = setS[m].compareTo(q)
- BinarySearch를 직접 구현하지 않아도 Arrays 클래스의 binarySearch() 메서드를 이용하여 문제를 풀수 있다. binarySearch() 메서드는 주어진 key가 존재한다면 그 key의 index 값을 반환한다.
- https://docs.oracle.com/javase/8/docs/api/java/util/Arrays.html
- https://www.acmicpc.net/problem/2295
- 식을 변형해서 풀 생각을 하지 못했다.
$A+B+C=X$ ->$A+B = X-C$ - 3중 for문을 돌아도 index 탐색 순서와 비교 로직에 따라서 제한 시간 내에도 풀 수 있다는 것을 배웠다. https://www.acmicpc.net/source/64583452
- https://www.acmicpc.net/problem/2470
- 배열의 index에 들어있는 값을 남겨야 할 때와 배열의 index를 남겨야 할 때를 잘 구분해자.
- TreeSet의 floor()와 ceiling() 메서드를 이용하면
$O(NlogN)$ 시간 복잡도로 근사값을 찾을 수 있다는 것을 배웠다.
- https://www.acmicpc.net/problem/10816
- Map의 메서드 중 getOrDefault()가 있다는 것을 학습했으면서도 정작 없는 값을 찾을 때 if문 내에서 getContainsKey() 메서드를 사용했다.
- Binary search를 경계값을 찾는데 응용할 수 있다는 것을 배웠다.
- 메모리가 넉넉하다면 1천만 이상 크기의 배열도 생생해서 인덱싱을 이용한 풀이도 시도해 볼 수 있다.
- https://www.acmicpc.net/problem/2805
- 처음에 주어진 수들의 평균을 구해서 parametric search의 시작 파라미터로 했는데 예제처럼 원하는 나무의 크기가 평균보다 작다면 조건을 만족할 수 있지만, 원하는 나무의 크기가 평균보다 크다면 조건을 만족할 수 없는 파라미터가 된다. 이것을 생각하지 못했다.
- https://www.acmicpc.net/problem/6236
- 경계값을 구할 때 최대값은 N = 1일 수 있으므로 N * 10000이 되어야 하지만 이런 케이스를 고려하지 못하고 단순하게 한번 출금하는 최대금액인 10000으로 설정하는 오류가 있었다.
- https://www.acmicpc.net/problem/2110
- 공유기 사이의 최소 거리와 최대거리를 정하는 경계값 계산에 오류가 있었다. 공유기 사이의 최소 거리는 1, 최대거리는 공유기를 최소 2개 설치할 수 있으므로 집의 마지막 위치 - 집의 최초 위치가 되어야 했다.
- https://www.acmicpc.net/problem/2512
- 배정된 예산을 계산할 때 문제를 잘못 이해하고 최소 금액을 N으로 설정하는 오류가 있었다. 각 지방의 예산 요청은 금액은 1~100,000 이하인데 이 값을 M과 혼동했다.
- https://www.acmicpc.net/problem/2343
- 블루레이 사이즈를 판정하는 로직에서 블루레이에 담을 수 없는 경우를 앞에서 모두 필터링 했다면 마지막에는
return true;
를 했어야 하는데return count >= maxBluRayCount;
를 해서 항상 false가 반환되는 오류가 있었다.
- https://www.acmicpc.net/problem/2792
- 나누어 떨어지는 경우와 나누어 떨어지지 않는 경우를 분류하기 위해 if - else 를 사용해도 되지만 간단한게
(color + divideCount - 1) / divideCount
와 같이 나눌 수 -1을 해서 더해주고,다시 나눌 수로 나눠주면 된다. - 문제 분석을 잘못해서 보석을 학생들에게 차례대로 나눠주는 오류가 있었다.
- https://www.acmicpc.net/problem/2143
- 부분합을 미리 구해놓는다는 아이디어를 떠올리지 못했고, 예제의 케이스만 부분합에 적용하는 오류가 있었다.
- https://www.acmicpc.net/problem/2003
- 경계값을 설정하는대 오류가 있었다. 머리속으로만 계산하지 말고 헷갈리면 종이에 그리는 습관을 들이자.
- https://www.acmicpc.net/problem/2470
- 이런 유형의 문제는 투 포인터를 적용하면 로직이 깔끔해진다.
- 정확히 떨어지는 값이 없을 수도 있으므로 초기값을 설정해 주어야 하는데 그 부분에 오류가 있었다.
- https://www.acmicpc.net/problem/1806
- 길이 N짜리 수열이라는 말을 제대로 이해하지 못하고 StringBuilder로 String으로 변환하여 자리수를 구한다는 아이디어를 구현했지만 이것은 답이 아니었다.
- 수의 개수를 초기값으로 설정 할 때 0으로 설정하고 Math.min()으로 답을 찾는 오류가 있었다.
- https://www.acmicpc.net/problem/2118
- 탑이 자리한 위치와 탑간의 거리를 구별해서 시각화 해야 해쓴ㄴ데 탑이 자리한 위치를 빼고 탑간의 거리만 시각화하는 오류로 인해서 로직을 도출해 내지 못했다.
- https://www.acmicpc.net/problem/17609
- 처음 로직은 두 포인터의 문자를 비교하여 다르면 왼쪽 문자와 오른쪽 문자를 하나씩 지워보고 맞는 방향으로 진행하는 로직을 짰는데 abbab와 같은 반례를 생각하지 못했기 때문이다. 이 케이스의 경우 앞으 a를 지우면 결과는 2가 나오지만 뒤의 b를 지우면 0이 나온다. 이런 반례를 생각하지 못했기 때문에 푸는데 3시간이 넘게 걸렸다.
- 결국 처음 로직을 폐기하고 재귀를 사용하여 풀었지만 한번만 더 생각했다면 시간을 허비하지 않고 풀 수 있는 문제였다.
- https://www.acmicpc.net/problem/15831
- 슬라이딩 윈도우를 유지하면서 현재 검은돌의 개수가 최대치 미만이면 nextIndex를 1 증가시키면서 nextIndex에 위치한 돌의 색상 개수를 1 증가시키고, 검은돌의 개수가 최대치를 초과했다면 startIndex의 돌 색상을 1 감소시켜서 풀어야 했는데, 슬라이딩 윈도우를 이동할 때마다 돌의 개수를 다시 세는 로직을 짜서 시간초과를 계속 받았다.
- https://www.acmicpc.net/problem/16472
- 문자열을 탐색하면서 다른 문자가 나올 때 count를 증가시키고, count의 수가 N을 넘어서면 start 위치를 이동시켰는데 start 위치를 이렇게 이동시키면 주어진 예제에서는 맞는 답이 나올 수 있으나 다른 반례에서는 틀린 답이 나올 수 있다. 주어진 예제 외에도 다양한 반례들로 테스트 해보는 연습을 꾸준히 하자.
- https://www.acmicpc.net/problem/14465
- 슬라딩 윈도우의 개념을 정확히 이해하지 못해서 인덱스가 바뀔 때마다 전체 구간의 합을 다시 계산하는 실수를 했다.
- 구간합을 미리 계산해 두고 구간합을 이용해서도 풀 수 있는 문제였다.
- https://www.acmicpc.net/problem/10025
- 좌표인 X = 0 ~ 1,000,000 인데, 좌우로 닿을 수 있는 거리인 K = 1 ~ 2,000,000인 문제였다. K가 X보다 큰 경우를 생각하지 못하고 예제 케이스만 보고 index을 구현했다가 IndexOutOfBoundsException 에러을 계속 만났다.
- https://www.acmicpc.net/problem/13422
- 슬라이딩 윈도우를 이용해서 비교 대상 구간합을 미리 구해 놓고 다음 구간합을 구할 때
prefixSum -= house[i - 1]
로 현재index - 1
을 해주어야 했는데 현재 index인 i를 빼주는 실수는 했다. - 환형 index 구조를 계산해야 한다면
(i + M - 1) % N
으로 계산해주면 된다. N = 10, i = 8, M = 3이라고 가정하면 8 + 3 - 1 = 11 이므로 11 % 10을 하면 1이 나와서 환형 index 구조를 이용할 수 있게 된다.
- https://www.acmicpc.net/problem/13144
- 중복을 체크하는 배열를 만들 때 수열의 수를 index로 삼겠다고 생각했으면서 정작 배열의 크기는 수열의 길이인 N으로 만드는 실수가 있었다.
1개 이상의 수를 뽑았을 때 같은 수가 여러 번 등장하지 않는 경우의 수
라는 조건을 이해하는데 시간이 오래 걸렸다.
- https://www.acmicpc.net/problem/1406
- 삽입 및 삭제가 많은 문제 유형이라 LinkedList를 이용하지 않으면 시간 초과를 받게 된다. 처음에 AyyarList를 이용하여 풀었고 계속 시간 초과를 받았다.
- ListIterator를 선언해 놓고 Iterator를 사용하지 않고 직접
list.add()
를 해서concurrentmodificationexception
이 발생하는 실수가 있었다.
- https://www.acmicpc.net/problem/15828
java.util.concurrent
패캐지의LinkedBlockingQueue
를 이용하여 버퍼사이즈를 제한할 수 있다는 것을 배웠다. 하지만 concurrent 패키지는 멀티스레드 환경에서 동시성 문제를 해결하기 위한 패키지이므로 일반적인 환경에서는 성능이 조금 느릴 수 있다.
- https://www.acmicpc.net/problem/5430
- R 명령이 입력되었을 때 시간 복잡도 계산을 하지 못하고 무조건 뒤집는 로직을 구현해서 시간 초과르 받았다.
- 숫자가 1자리 숫자만 있는 것이 아닌데 역순 출력해야 하는 경우 StringBuilder로 변환해 놓고 이걸 sb.reverse()로 뒤집었는데 1자리 숫자가 아니라면 당연히 틀린 결과가 나온다. 이 생각의 오류를 찾는데 시간이 오래 걸렸다.
- StringTokenizer의 delimeter로
"[,]"
3개 문자를 모두 입력한다는 아이디어를 떠올리지 못하고"[]"
문자를 먼저 remove 한다는 아이디어에 매몰되었다.
- https://www.acmicpc.net/problem/9012
- 괄호의 종류가 하나이므로 count를 이용하여 풀어볼 수 있다. ex) '(' 입력시 count 증가, ')' 입력시 count 감소.
- https://www.acmicpc.net/problem/10828
- 복잡한 구현 문제가 아니므로 굳이 스택을 사용하지 않고 배열과 index 조작만으로도 풀이할 수 있다.
- https://www.acmicpc.net/problem/10799
- 쇠막대기의 시작 위치와 끝위치를 index를 이용해서 풀어볼 생각을 하지 못했다. 또한 레이저의 개수 합을 구간합을 이용해서 미리 구한다는 아이디어를 떠올리지 못했다.
- https://www.acmicpc.net/problem/5397
- 커서의 위치를 기준으로 List + Itrator 조합이 아닌 스택 2개를 이용하여 풀이하는 방법을 배웠다.
- https://www.acmicpc.net/problem/16120
- PPAP 문자열이 나오면 계속 P로 치환을 하고 마지막에 P 문자열이 남는 것으 체크해야 하는데 PPAP를 한번만 치환하는 아이디어의 오류가 있었다.
- https://www.acmicpc.net/problem/2841
- 처음에 2차원 배열로 푸는 아이디어를 떠올렸는데
Deque<Integer>[] pushed
와 같은 형태로 Collection 배열을 만들 수 있다.
- https://www.acmicpc.net/problem/17298
- 예전에 한번 풀었는데 저장된 배열의 오른쪽에서 왼쪽으로 탐색하면서 오큰수 후보만 스택에 남긴다는 아이디어를 바로 떠올리지 못했다.
- https://www.acmicpc.net/problem/1935
- 연산자 입력이 들어오면 그때까지 입력된 피연산자만 연산해야 하는데 뒤에 입력 받은 피연산자까지 모두 합쳐서 연산하는 생각의 오류가 있어서 문제의 요구사항 이해가 늦게 되었다.
- https://www.acmicpc.net/problem/17413
- 뒤집어야 하는 구간을 찾을 수 있다면 스택을 사용하지 않아도 거꾸로 기록할 수 있다.
- 복합 조건에서 배열의 index 관리에 미숙함이 있었다.
- https://www.acmicpc.net/problem/6198
- 백준 19298 오큰수 문제와 기준점은 같지만 왼쪽 -> 오른쪽으로 진행하면서 옥상을 볼 수 있는 빌딩의 수를 세어야 하는데 최초에 오큰수 문제와 같이 역순으로 접근하는 아이디어의 오류가 있었다.
- https://www.acmicpc.net/problem/2812
- 자리값이 큰 숫자를 삭제하여 그 자리에 더 큰 숫자가 올 수 있다면 삭제하는 것이 이득이다. 앞에서부터 K개를 삭제하여 내림차순이 되도록 부분 수열을 만들고, 더 삭제할 수 없으면 그대로 사용한다는 아이디어를 떠올리지 못했다.
- https://www.acmicpc.net/problem/15657
- 비내림차순과 오름차순의 차이점을 파악하지 못했다. 비내림차순이란 재귀호출 시 현재의 자기 자신을 포함할지 여부가 오름차순과 다르다.
백준 1260 DFS와 BFS: https://www.acmicpc.net/problem/1260
- 인접 리스트 방식으로 구현할 때 배열의 생성은 N + 1개로 해놓고 ArraysList의 sort는 N개까지만 하는 오류가 있었다.
- https://www.acmicpc.net/problem/2606
- Queue + LinkedList 조합이 Deque + ArrayDeque 조합보다 근소하게 빠른 성능을 보였다.
- https://www.acmicpc.net/problem/2573
- 빙산이 있는 좌표만 별도의 클래스로 분리하여 관리한다는 아이디어를 떠올리지 못했다.
- https://www.acmicpc.net/problem/1941
- 2차원 배열을 1차원 배열로 치환해서 백트래킹 할 수 있다는 것을 배웠다.
- https://www.acmicpc.net/problem/1759
- String input을 char[] 배열로 만들 때 static으로 미리 변수 선언을 해놓은 것을 고려하지 않고 코드를 작성하는 생각의 오류가 있었다.
-
완전 탐색 알고리즘을 사용해야 풀 수 있는 문제 유형이었다. 재귀 함수 호출 포인트를 잡는 연습을 많이 해야겠다.
-
완전 탐색 알고리즘의 시간 복잡도와 공간 복잡도 정리
중복 순서 시간 복잡도 공간 복잡도 YES YES $\large O(N^M)$ $O(M)$ NO YES $\large O(^N_MP=O \left( \frac{N!}{(N-M)!} \right)$ $O(M)$ YES NO $\large O(N^M)$ 보단 작음$O(M)$ NO NO $\large O(^N_MP=O \left( \frac{N!}{M!(N-M)!} \right)$ $O(M)$
- https://www.acmicpc.net/problem/14888
- 숫자의 순서가 바뀌면 안되고, 연산자의 순서는 바뀌면서 연산 결과 최대 값과 최소 값을 찾아야 하는데 연산자를 어떻게 섞어야 할지 아이디어를 떠올리지 못해서 결국 강의를 보고 말았다. 알고리즘이 단시일내에 해결되는 게 아닌데 마음만 너무 앞서 갔던 것 같다. 기본으로 돌아가서 내가 어떤 부분을 모르고 어떤 부분에 강점이 있는지 차분하게 점검하는 시간을 가져야 겠다.