Notice
Recent Posts
Recent Comments
Link
일 | 월 | 화 | 수 | 목 | 금 | 토 |
---|---|---|---|---|---|---|
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 |
Tags
- python
- 피보나치 함수
- 알고리즘
- 파이썬
- BFS: Shortest Reach in a Graph
- Recursion: Davis' Staircase
- 매칭점수
- candies
- hackerrank
- DFS: Connected Cell in a Grid
- 격파르타 합격후기
- 프로그래머스
- Special String Again
- Find the nearest clone
- 격파르타 장점
- programmers
- Common Child
- Max Array Sum
- 해커랭크
- Roads and Libraries
- 머신러닝
- 구슬탈출2
- 야근지수
- [sqld]자격증합격
- Reverse Shuffle Merge
- Algorithm
- 격파르타 후기
- 백준
- Interview Preparation Kit
- 코딩테스트
Archives
- Today
- Total
Archive
[Hackerrank] [Hard] Merge Sort: Counting Inversions 본문
https://www.hackerrank.com/challenges/ctci-merge-sort/problem
풀이
- 예제처럼 bubble sort로 구현하면 timeout 발생
- merge sort로 구현해도 timeout 발생함
- merge 부분에서 list의 append함수를 지역 변수로 사용하는게 빠름.
- 필요한 정보들( len(list) 등)을 미리 변수에 할당하고 사용하는게 빠름.
- 가운데 index를 계산해 둘 씩 나누다가, 합칠 때 inversion 경우를 count함.
- i < j 일 때, arr[i] < arr[j]라면 넘어가고, arr[i] > arr[j]일 때 계산
- left_lst_len - i를 더해줌.
- [1, 4, 7] / [3, 5, 8 ]을 sorted_lst에 merge하는 상황에서 오른쪽 리스트에서 3을 append하는 경우를 생각.
- merge sort의 중간 과정이기 때문에 left_lst, right_lst는 항상 정렬되어 있음.
- sorted_lst = [1] , i = 1(4에 대한 index), j = 0(3에 대한 index)
- 예제처럼 인접한 원소끼리 값을 swap한다면, 2번을 해야 함.
- [1, 4, 7, 3, 5, 8] -> [1, 4, 3, 7, 5, 8] -> [1, 3, 4, 7, 5, 8]
- 하지만 swap 수를 바로 구할 수 있는데, left_lst의 전체 길이에서, sorted_lst에 들어간 원소 개수인 i를 빼주면 됨.
- i는 왼쪽 리스트에서 값을 비교하기 위한 index 정보, sorted_lst에 들어간 원소 개수에 대한 정보를 의미함.
- [4, 7] 이 되고, right_idx[0] = 3보다 큰 수가 2개가 되기 때문에 2번 값을 바꿔주면 답을 구할 수 있음.
- j의 정보는 상관없는게, arr [ j ]보다 큰 수가 left_lst에서 몇 개 있는지만 확인하면 되기 때문.
- (참고) left_lst [ i : ]의 길이를 구해도 되지만, timeout 발생함. len(left_lst[i : ])
- [1, 4, 7] / [3, 5, 8 ]을 sorted_lst에 merge하는 상황에서 오른쪽 리스트에서 3을 append하는 경우를 생각.
def sortAndCountInversion(arr):
count_inversion = merge_sort(arr)
return count_inversion[1]
def merge_sort(lst):
if len(lst) <= 1:
return lst, 0
n = len(lst)
mid = n//2
left_lst, l_cnt = merge_sort(lst[:mid])
right_lst, r_cnt = merge_sort(lst[mid:])
sorted_lst, cnt = merge(left_lst, right_lst)
return sorted_lst, l_cnt + r_cnt + cnt
def merge(left_lst, right_lst):
sorted_lst = []
left_lst_len = len(left_lst)
right_lst_len = len(right_lst)
# 이렇게 안하면 timeout 발생.
lst_append = sorted_lst.append
i, j = 0, 0
cnt = 0
while i < left_lst_len and j < right_lst_len:
if left_lst[i] <= right_lst[j]:
lst_append(left_lst[i])
i += 1
else:
lst_append(right_lst[j])
j += 1
# len(left_lst[i:]) timeout 발생
cnt += left_lst_len - i
sorted_lst += left_lst[i :]
sorted_lst += right_lst[j:]
return sorted_lst, cnt
'공부 > Algorithm' 카테고리의 다른 글
[Hackerrank] [Medium] Special String Again (0) | 2020.08.28 |
---|---|
[Hackerrank] [Medium] Sherlock and the Valid String (0) | 2020.08.27 |
[Hackerrank] [Medium] Fraudulent Activity Notifications (0) | 2020.08.24 |
[Hackrrank] [Medium] Count Triplets (0) | 2020.08.23 |
[Hackerrank] [Medium] Sherlock and Anagrams (0) | 2020.08.22 |
Comments