Indexed Priority Queue - 우선순위 큐의 임의 원소 삭제 (2) : 최적화
A형 필수 알고리즘을 체계적으로 배우고 싶다면? (인프런 바로가기)
참고
- 우선순위 큐 응용 (1) - 두 개의 heap을 이용하여 중앙값 찾기
- 우선순위 큐 응용 (2) - 최댓값, 최솟값 동시에 관리하기
- 우선순위 큐 임의 원소 삭제 최적화
Indexed Priority Queue, Heap - 삭제 구현, 최적화
이전 글에서는 우선순위 큐의 임의 원소 삭제에 O(NlogN)의 비용을 O(N) + O(logN) ≒ O(N)으로 줄였다.
이제 memoization을 이용하여 O(N) + O(logN)을 O(1) + O(logN) ≒ O(logN)으로 줄여보자.
아이디어는 다음과 같다.
deleteId 함수에서 delhn = findIndex(id);를 delhn = heapIdx[id]; 로 변경하면 된다.
heapIdx는 int 배열이며, heapIdx[id] = id의 현재 heap index를 저장하고 있다.
typedef struct st
{
int value;
int id;
}HEAP;
HEAP heap[16];
int heapIdx[16]; /* id가 heap의 어느 위치에 있는지 저장할 배열 */
int hn;
heapIdx에 현재 id가 heap의 어디에 위치해있는지 저장/갱신하기 위해 push 함수를 다음처럼 바꾼다.
void push(HEAP* heap, int& hn, int value, int id, int* heapIdx)
{
register HEAP tmp;
heap[++hn].value = value;
heap[hn].id = id;
heapIdx[id] = hn; /* id에 현재 hn을 저장 */
for (register int i = hn; i > 1; i /= 2)
{
if (isMin(heap[i / 2], heap[i])) break;
tmp = heap[i / 2];
heap[i / 2] = heap[i];
heap[i] = tmp;
/* heapIdx 갱신 */
heapIdx[heap[i / 2].id] = i / 2;
heapIdx[heap[i].id] = i;
}
}
최초로 heapIdx[id]는 현재 heap에 push에 의해 결정된 hn이 된다.
현재 heap[i / 2]의 id는 i / 2가 될 것이고, heap[i]의 id는 i가 되므로 위와 같이 갱신하면 된다.
마찬가지로 pop도 아래처럼 갱신을 추가하자.
HEAP pop(HEAP* heap, int& hn, int* heapIdx)
{
register HEAP tmp, ret;
ret = heap[1];
heap[1] = heap[hn];
heap[hn].value = 0x7fff0000;
heap[hn--].id = 0x7fff0000;
for (register int i = 1; i * 2 <= hn;)
{
if (isMin(heap[i], heap[i * 2]) && isMin(heap[i], heap[i * 2 + 1])) break;
else if (isMin(heap[i * 2], heap[i * 2 + 1]))
{
tmp = heap[i * 2];
heap[i * 2] = heap[i];
heap[i] = tmp;
/* heapIdx 갱신 */
heapIdx[heap[i].id] = i;
heapIdx[heap[i * 2].id] = i * 2;
i = i * 2;
}
else
{
tmp = heap[i * 2 + 1];
heap[i * 2 + 1] = heap[i];
heap[i] = tmp;
/* heapIdx 갱신 */
heapIdx[heap[i].id] = i;
heapIdx[heap[i * 2 + 1].id] = i * 2 + 1;
i = i * 2 + 1;
}
}
return ret;
}
pop과 push의 함수에 int* heapIdx가 추가하는 것을 잊지 말자.
여기까지 구현했다면, 이전 글의 input 그대로 다시 pop해보자.
HEAP pop(HEAP* heap, int& hn, int* heapIdx);
void push(HEAP* heap, int& hn, int value, int id, int* heapIdx);
int main(void)
{
int input[] = { 6, 5, 3, 5, 2, 3, 4, 5, 6, 7, 2, 5, 6, 7, 0 }; /* 점수 input */
int size = sizeof(input) / sizeof(int);
for (int i = 0; i < size; i++) push(heap, hn, input[i], i + 1);
return 0;
}
그러면 아래와 같은 heap과 heapIdx가 완성된다.
id = 1부터 시작하므로 heapIdx[0] = 0이다.
위의 그림에서 heapIdx[3] = 5임을 알 수 있고, 실제 그림에서도 5번째 heap에 id = 3이 존재한다.
다른 id들도 heapIdx에 자신의 위치가 제대로 저장되어있는지 직접 확인해보자.
이제 deleteId를 수정하자.
delhn은 heapIdx[id]로 O(1)만에 찾을 수 있다.
그리고 deleteId의 delhn부터 heap을 갱신하는 코드에도 heapIdx 갱신을 추가하면 된다.
// int findIndex(int id);
int deleteId(HEAP* heap, int& hn, int id, int* heapIdx)
{
HEAP tmp;
int ret;
int delhn = heapIdx[id]; // findIndex(id);
ret = heap[delhn].value;
heap[delhn].value = -1;
for (int i = delhn; i > 1; i /= 2)
{
if (isMin(heap[i / 2], heap[i])) break;
tmp = heap[i];
heap[i] = heap[i / 2];
heap[i / 2] = tmp;
/* heapIdx 갱신 */
heapIdx[heap[i].id] = i;
heapIdx[heap[i / 2].id] = i / 2;
}
pop(heap, hn, heapIdx);
return ret;
}
바뀐 deleteId로 id = 3을 제거하면 아래와 같은 결과가 나온다. (이전과 같은 결과이다.)
heapIdx[3] = 1로 가장 최근에 갱신된 내용이 남아있게 된다.
id = 3은 존재하지 않으므로, heapIdx[3]을 굳이 고칠 필요는 없다.
(문제 조건에 따라 id = 3이 존재하는지 묻는다면, 0이나 -1로 바꾸는 코드를 추가하면 된다.)
갱신된 heap도 여전히 heapIdx[id]가 id의 위치를 잘 가르키는지 직접 확인해보자.
최종 코드는 아래와 같다.
#include <stdio.h>
typedef struct st
{
int value;
int id;
}HEAP;
HEAP heap[16];
int heapIdx[16]; /* id가 heap의 어느 위치에 있는지 저장할 배열 */
int hn;
int isMin(HEAP a, HEAP b)
{
if (a.value < b.value) return 1;
else if (a.value == b.value && a.id < b.id) return 1;
return 0;
}
void output(HEAP* heap)
{
for (int i = 1; i <= 15; i++)
{
printf("%d (id : %d) ", heap[i].value, heap[i].id);
if ((i & (i + 1)) == 0) putchar('\n');
}
putchar('\n');
}
void outputHeapIdx(int* heapIdx)
{
for (int i = 1; i <= 15; i++) printf("%d ", heapIdx[i]);
putchar('\n');
}
HEAP pop(HEAP* heap, int& hn, int* heapIdx)
{
register HEAP tmp, ret;
ret = heap[1];
heap[1] = heap[hn];
heap[hn].value = 0x7fff0000;
heap[hn--].id = 0x7fff0000;
for (register int i = 1; i * 2 <= hn;)
{
if (isMin(heap[i], heap[i * 2]) && isMin(heap[i], heap[i * 2 + 1])) break;
else if (isMin(heap[i * 2], heap[i * 2 + 1]))
{
tmp = heap[i * 2];
heap[i * 2] = heap[i];
heap[i] = tmp;
heapIdx[heap[i].id] = i;
heapIdx[heap[i * 2].id] = i * 2;
i = i * 2;
}
else
{
tmp = heap[i * 2 + 1];
heap[i * 2 + 1] = heap[i];
heap[i] = tmp;
heapIdx[heap[i].id] = i;
heapIdx[heap[i * 2 + 1].id] = i * 2 + 1;
i = i * 2 + 1;
}
}
return ret;
}
void push(HEAP* heap, int& hn, int value, int id, int* heapIdx)
{
register HEAP tmp;
heap[++hn].value = value;
heap[hn].id = id;
heapIdx[id] = hn;
for (register int i = hn; i > 1; i /= 2)
{
if (isMin(heap[i / 2], heap[i])) break;
tmp = heap[i / 2];
heap[i / 2] = heap[i];
heap[i] = tmp;
heapIdx[heap[i].id] = i;
heapIdx[heap[i / 2].id] = i / 2;
}
}
int deleteId(HEAP* heap, int& hn, int id, int* heapIdx)
{
HEAP tmp;
int ret;
int delhn = heapIdx[id]; // findIndex(id);
ret = heap[delhn].value;
heap[delhn].value = -1;
for (int i = delhn; i > 1; i /= 2)
{
if (isMin(heap[i / 2], heap[i])) break;
tmp = heap[i];
heap[i] = heap[i / 2];
heap[i / 2] = tmp;
heapIdx[heap[i].id] = i;
heapIdx[heap[i / 2].id] = i / 2;
}
pop(heap, hn, heapIdx);
return ret;
}
int main(void)
{
int input[] = { 6, 5, 3, 5, 2, 3, 4, 5, 6, 7, 2, 5, 6, 7, 0 };
int size = sizeof(input) / sizeof(int);
for (int i = 0; i < size; i++) push(heap, hn, input[i], i + 1, heapIdx);
printf("%d\n", deleteId(heap, hn, 3, heapIdx));
output(heap); printf("\n");
outputHeapIdx(heapIdx);
return 0;
}