알고리즘/알고리즘 기초공부
동빈나 2021 이코테_5. 이진 탐색
이숭간
2021. 2. 10. 13:05
728x90
- 순차탐색 : 리스트 안에 있는 특정한 데이터를 찾기위해 앞에서부터 하나씩 확인
- 이진탐색 : 정렬되어 있는 리스트에서 탐색범위를 절반씩 좁혀가며 데이터 탐색 ( 시작점, 끝점, 중간점을 이용하여 탐색범위를 설정한다.)
단계마다 절반씩 줄어드는 알고리즘의 시간복잡도는 보통 로그N
<예제문제1>
내코드 _ 잘못됨 ( 완전탐색이므로 시간초과남 )
n,m = map(int, input().split())
input_list = list(map(int, input().split()))
for i in range(max(input_list), 0, -1):
count = 0
for j in input_list:
count += max(0, j-i)
if count >= m:
print(i)
break
내코드2 _ 이진탐색으로 다시 풀어봄
import sys
sys.setrecursionlimit(10**6)
n,m = map(int, input().split())
input_list = list(map(int, input().split()))
end = max(input_list)
start = 0
count = 0
result = []
def binary_search(input_list, start, end):
if start > end:
return None
count = 0
mid = start + end //2
for j in input_list:
count += max(0, j-mid)
if count >= m: #잘린떡이 충분한경우
result.append(count)
return binary_search(input_list, mid+1, end)
else: # 잘린떡이 모자란경우 ( 더 짧게 잘라야함)
return binary_search(input_list, start, end-1)
binary_search(input_list, start, end)
print(result[-1])
매우 큰 탐색범위를 보면 먼저 이진탐색을 떠올려야함
동빈나님 코드
< 예제문제2 _ 정렬된 배열에서 특정수의 개수 구하기 >
처음 전체탐색 범위에 대해서 이진탐색을 두번 수행하여 1번 이진탐색은 첫번째위치, 2번 이진탐색은 마지막위치를 찾아서 두 위치간 차이를 구하면 됨
실제 이진탐색 코드를 직접 작성해서 풀수도 있고, 표준라이브러리로도 구현할 수 있다.
동빈나님 코드: (표준라이브러리사용)
#count_by_range()는 여러 코딩테스트에서 유용하게 사용될수 있으므로 팁노트에 저장하여 사용해놓으면 좋음