티스토리 뷰
[주제]
- 앞서 구현한 힙 구조에서 일부분 업그레이드하기
[변경되는 부분]
- 프로그래머가 우선순위 판단 기준을 설정해놓기
- 우선순위 비교 함수를 만들어서 우선순위 비교하기
[중요]
- 우선순위 비교 함수의 기능
1. 첫 번째 인자의 우선순위가 높으면: '0'보다 큰 값을 반환
2. 두 번째 인자의 우선순위가 높으면: '0'보다 작은 값을 반환
3. 각 인자의 우선순위가 동일하면: '0'을 반환
[소스]
■ UsefulHeap.h
#ifndef SIMPLEHEAP_H_ #define SIMPLEHEAP_H_ #define TRUE 1 #define FALSE 0 #define HEAP_LEN 100 /* * 'Heap' 구조체에 통합 */ /* typedef struct _heapElem { int pr; char data; } HeapElem; */ typedef int PriorityComp(char d1, char d2); // 인자값 1과 2의 우선순위를 비교할 포인터 함수 typedef struct _heap { PriorityComp * comp; // 우선순위 비교 함수를 가리키는 'comp' 선언 int numOfData; // HeapElem heapArr[HEAP_LEN]; // (구)데이터를 저장했던 구조체 삭제 char heapArr[HEAP_LEN]; // (신)'char' 자료형 구조체로만 선언 } Heap; // void HeapInit(Heap * ph); // (구)힙 초기화 void HeapInit(Heap * ph, PriorityComp pc); // (신)인자값으로 우선순위 비교 함수 추가 int HIsEmpty(Heap * ph); // 힙에 데이터가 비워져 있는지 확인 // void HInsert(Heap * ph, char data, int pr); // (구)새로운 노드 삽입 void HInsert(Heap * ph, char data); // (신)데이터 자체로 우선순위가 설정되어 있기 때문에 인자값 'pr' 사용 안 함 char HDelete(Heap * ph); // 루트노드 삭제 #endif | cs |
■ UsefulHeap.c
#include "UsefulHeap.h" /* // (구)힙 초기화 void HeapInit(Heap * ph) { ph->numOfData = 0; } */ /* * 우선순위 비교에 사용할 'comp' 초기화 */ // (신)힙 초기화 void HeapInit(Heap * ph, PriorityComp pc) { ph->numOfData = 0; ph->comp = pc; } // 힙에 데이터가 비워져 있는지 확인 int HIsEmpty(Heap * ph) { if (ph->numOfData == 0) { return TRUE; } else { return FALSE; } } // 현재 노드의 부모노드 인덱스 찾기 int GetParentIDX(int idx) { return idx / 2; } // 현재 노드의 왼쪽 자식노드 인덱스 찾기 int GetLChildIDX(int idx) { return idx * 2; } // 현재 노드의 오른쪽 자식노드 인덱스 찾기 int GetRChildIDX(int idx) { return GetLChildIDX(idx) + 1; } /* * [두 개의 자식노드 중 높은 우선순위의 노드 인덱스 찾기] * - 왼쪽 자식노드 번호가 전체 노드개수 보다 높을 경우: 자식노드가 없는 경우를 의미 * - 왼쪽 자식노드 번호가 전체 노드개수와 동일한 경우: 왼쪽 자식노드가 존재하며, 마지막 노드를 의미 * - 자식노드가 양쪽에 존재하고, 왼쪽 자식노드의 우선순위 숫자가 * 오른쪽 자식노드의 우선순위 숫자보다 높을 경우: 오른쪽 자식노드의 우선순위가 높다는 것을 의미 * - 자식노드가 양쪽에 존재하고, 우선순위 숫자가 서로 같거나 왼쪽 자식노드가 낮을 경우: * 왼쪽 자식노드의 우선순위가 높다는 것을 의미 * * [변경된 부분] * - 자식노드가 양쪽에 존재하고, 왼쪽 자식노드와 오른쪽 자식노드를 비교했을 때 * '0 미만'이면 오른쪽 자식노드가 우선순위가 높은 것으로 판단 * '0 또는 이상'이면 왼쪽 자식노드가 우선순위가 높은 것으로 판단 */ int GetHiPriChildIDX(Heap * ph, int idx) { if (GetLChildIDX(idx) > ph->numOfData){ return 0; } else if (GetLChildIDX(idx) == ph->numOfData) { return GetLChildIDX(idx); } else { // if (ph->heapArr[GetLChildIDX(idx)].pr // > ph->heapArr[GetRChildIDX(idx)].pr) { if(ph->comp(ph->heapArr[GetLChildIDX(idx)], ph->heapArr[GetRChildIDX(idx)]) < 0) { return GetRChildIDX(idx); } else { return GetLChildIDX(idx); } } } /* * [새로운 노드 생성 및 우선순위에 맞는 자리 찾아가기] * - 'idx'에 마지막 노드의 위치를 저장 * - 생성된 노드에 우선순위 숫자와 데이터 저장 * - 생성된 노드가 루트 노드가 아닐 경우 다음을 반복 * - 생성된 노드의 우선순위 숫자가 부모노드의 우선순위 숫자보다 낮을 경우: * 생성된 노드가 우선순위가 높다는 것을 의미 -> 부모노드와 위치를 교체 * - 우선순위 숫자가 같거나 높을 경우 반복 종료 * - 데이터 개수 1개 추가 * * [변경된 부분] * - 'HeapElem'은 사용 안 함 * - 현재 노드와 부모노드의 우선순위를 비교 했을 때 * '0 초과'이면 부모노드를 배치 */ // void HInsert(Heap * ph, char data, int pr) { void HInsert(Heap * ph, char data) { int idx = ph->numOfData + 1; // HeapElem nelem = {pr, data}; while (idx != 1) { // if (pr < (ph->heapArr[GetParentIDX(idx)].pr)) { if (ph->comp(data, ph->heapArr[GetParentIDX(idx)]) > 0) { ph->heapArr[idx] = ph->heapArr[GetParentIDX(idx)]; idx = GetParentIDX(idx); } else { break; } } ph->heapArr[idx] = data; ph->numOfData += 1; } /* * [루트노드 삭제 및 우선순위에 맞는 자리 찾아가기] * - 루트노드의 데이터는 반환될 'retData'에 저장 * - 마지막 노드는 루트노드 자리에 옮기지 않고 'lastElem'을 마지막 노드로 임시 설정하여 우선순위 비교 * - 우선순위가 높은 자식노드와 비교하여 우선순위가 낮으면 위치를 교체해가며 아래로 내려감 * - 우선순위가 더 높거나 더 이상 아래로 내려갈 수 없으면 최종적으로 위치가 고정됨 * - 노드 전체개수를 '1' 감소하고 루트노드의 데이터는 반환 * * [변경된 부분] * - 마지막 노드와 자식노드의 우선순위를 비교 했을 때 * '0 이상'이면 비교 정지 */ char HDelete(Heap * ph) { // char retData = (ph->heapArr[1]).data; char retData = (ph->heapArr[1]); // HeapElem lastElem = ph->heapArr[ph->numOfData]; char lastElem = ph->heapArr[ph->numOfData]; int parentIdx = 1; int childIdx; while(childIdx = GetHiPriChildIDX(ph, parentIdx)) { // if(lastElem.pr <= ph->heapArr[childIdx].pr) if (ph->comp(lastElem, ph->heapArr[childIdx]) >= 0) { break; } ph->heapArr[parentIdx] = ph->heapArr[childIdx]; parentIdx = childIdx; } ph->heapArr[parentIdx] = lastElem; ph->numOfData -= 1; return retData; } | cs |
■ UsefulHeapMain.c
#include <stdio.h> #include "UsefulHeap.h" // 우선순위 비교 함수 int DataPriorityComp(char ch1, char ch2) { return ch2 - ch1; } int main(void) { Heap heap; // HeapInit(&heap); HeapInit(&heap, DataPriorityComp); HInsert(&heap, 'A'); HInsert(&heap, 'B'); HInsert(&heap, 'C'); HInsert(&heap, 'D'); HInsert(&heap, 'C'); HInsert(&heap, 'B'); while(!HIsEmpty(&heap)) printf("%c \n", HDelete(&heap)); return 0; } | cs |
■ 실행결과
■ 첨부파일
'자료구조' 카테고리의 다른 글
[ch 10-1] 단순한 정렬 알고리즘_2 (선택 정렬) (0) | 2016.05.10 |
---|---|
[ch 10-1] 단순한 정렬 알고리즘_1(버블 정렬) (0) | 2016.05.09 |
[ch 09-2] 힙의 구현과 우선순위 큐의 완성_2(구현) (0) | 2016.05.02 |
[ch 09-2] 힙의 구현과 우선순위 큐의 완성_1(추가과정, 삭제과정 개념) (0) | 2016.04.21 |
[ch 09-1] 우선순위 큐의 이해 (0) | 2016.04.21 |
댓글
공지사항
최근에 올라온 글
최근에 달린 댓글
- Total
- Today
- Yesterday
링크
TAG
- 왕초보 영어회화 100일의 기적
- 응용
- java
- 정렬
- 영어
- 교육센터
- 문법
- System
- 작문
- ADODB
- VCL
- 계산기
- RA
- 독해
- 일기
- 알고리즘
- 설명
- wfd
- tdataset
- 자료구조
- SWT
- Reference
- 스택
- 말하기
- 대상
- SysUtils
- 여행영어 100일의 기적
- 상황
- Pte
- Delphi
일 | 월 | 화 | 수 | 목 | 금 | 토 |
---|---|---|---|---|---|---|
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 |
글 보관함