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
- 사이드프로젝트
- defee
- 백준
- Flutter
- 큐
- userprog
- HTML
- 스택
- 오블완
- 크래프톤 정글
- 리액트
- Java
- 코드트리
- 티스토리챌린지
- TiL
- 소켓
- JavaScript
- CSS
- 자바스크립트
- 알고리즘
- 자바
- Vue.js
- 나만무
- pintos
- 크래프톤정글
- 모션비트
- 핀토스
- corou
- 시스템콜
- 4기
Archives
- Today
- Total
미새문지
크래프톤 정글 week05, day31 - rb트리 구현 코드 본문
728x90
다 완성 못하고 발표하기엔 너무 아쉬워서 밤새면서 버텼다. 팀원 한 분은 삽입, 회전, minmax를 하셨고 한 분은 전체적인 rb트리 틀과 배열을 맡았다. 본인은 탐색과 삭제에 연관되있는 함수들을 구현했다.
rb트리 생성
// 트리 생성
rbtree *new_rbtree(void)
{
rbtree *p = (rbtree *)calloc(1, sizeof(rbtree));
if (!p)
{
// 메모리 할당 실패 처리
return NULL;
}
// 센티넬(가드) 노드를 생성하고 검은색으로 설정
p->nil = (node_t *)calloc(1, sizeof(node_t));
if (!p->nil)
{
// 메모리 할당 실패 처리
free(p); // 이미 할당된 p 메모리 해제
return NULL;
}
p->nil->color = RBTREE_BLACK;
p->root = p->nil;
return p;
}
삭제 시의 후위 순회
void delete_postorder(rbtree *t, node_t *currentNode)
{
// 만약 현재 노드가 nil이 아니라면
if (currentNode != t->nil)
{
// 왼쪽, 오른쪽, 본인 순서인 후위 순회
delete_postorder(t, currentNode->left);
delete_postorder(t, currentNode->right);
free(currentNode);
}
}
rb트리 삭제
void delete_rbtree(rbtree *t)
{
// 모든 노드를 순회하면서 메모리 해제 필요
// 후위 순회 방식을 사용해 자식 노드부터 메모리 해제 후, 루트 노드 해제
delete_postorder(t, t->root);
free(t->nil);
free(t);
}
왼쪽으로 회전하는 함수
// x x
// / --> \ .
// y y
void rbtree_left_rotate(rbtree *t, node_t *x)
{
node_t *y = x->right;
x->right = y->left; // y의 왼쪽 서브트리를 x의 오른쪽 서브 트리로 옮기기
if (y->left != t->nil) // 만약 y의 왼쪽 자식이 nil이 아니라면
{
y->left->parent = x; // y의 왼쪽 자식의 부모노드에 x값을 넣기
}
y->parent = x->parent; // x의 부모를 y에 연결
if (x->parent == t->nil) // 만약 x의 부모노드가 nil이 아니라면
{
t->root = y; // t의 루트 노드에 y를 넣기
}
else if (x == x->parent->left) // 그게 아니라 x가 x의 부모노드의 왼쪽 자식 값과 같으면
{
x->parent->left = y; // x의 부모노드의 왼쪽 자식에 y를 넣기
}
else
{
x->parent->right = y; // x의 부모노드의 오른쪽 자식에 y를 넣기
}
y->left = x; // x를 y의 왼쪽으로 놓기
x->parent = y;
}
오른쪽으로 회전하는 함수(왼쪽 회전 함수와 반대로 작성)
// x x
// / <-- \ .
// y y
void rbtree_right_rotate(rbtree *t, node_t *x)
{
node_t *y = x->left;
x->left = y->right;
if (y->right != t->nil)
{
y->right->parent = x;
}
y->parent = x->parent;
if (x->parent == t->nil)
{
t->root = y;
}
else if (x == x->parent->right)
{
x->parent->right = y;
}
else
{
x->parent->left = y;
}
y->right = x;
x->parent = y;
}
삽입 시 정렬
void rbtree_insert_fixup(rbtree *t, node_t *newNode)
{
// 삽입한 노드부터 루트 노드까지 거슬러 올라가며 다음과 같은 경우를 고려
while (newNode->parent->color == RBTREE_RED)
{
// 경우 1: 새로운 노드의 부모 노드가 조부모 노드의 왼쪽 자식인 경우
if (newNode->parent == newNode->parent->parent->left)
{
// 조부모 노드, 삼촌 노드(부모 노드의 형제) 정의
node_t *grandParent = newNode->parent->parent;
node_t *uncle = grandParent->right;
// 삼촌 노드 (부모 노드의 형제)가 빨간색인 경우:
if (uncle->color == RBTREE_RED)
{
// 부모 노드와 삼촌 노드의 색깔을 빨간색에서 검은색으로 변경
newNode->parent->color = RBTREE_BLACK;
uncle->color = RBTREE_BLACK;
// 조부모 노드의 색깔을 검은색에서 빨간색으로 변경
grandParent->color = RBTREE_RED;
newNode = grandParent;
}
// 삼촌 노드가 검은색인 경우:
else
{
// 새로운 노드가 부모 노드의 오른쪽 자식인 경우에만
if (newNode == newNode->parent->right)
{
newNode = newNode->parent;
노드 삽입
// 트리에 새로운 키를 가진 노드를 삽입하는 함수
node_t *rbtree_insert(rbtree *t, const key_t key)
{
// 일반 이진 탐색 트리처럼 노드 삽입
node_t *currentNode = t->root; // 루트 노드
node_t *parentNode = t->nil; // 추후 부모가 될 노드
// 루트 노드부터 내려가며 새로 노드가 삽입될 위치 찾기
while (currentNode != t->nil)
{
parentNode = currentNode;
if (key < currentNode->key)
{
currentNode = currentNode->left;
}
else
{
currentNode = currentNode->right;
}
}
// 받은 key 값을 가진 노드 추가
node_t *newNode = (node_t *)calloc(1, sizeof(node_t));
if (!newNode)
{
// 메모리 할당 실패 처리
return NULL;
}
// 삽입될 노드의 값 설정
newNode->parent = parentNode;
newNode->key = key;
// 노드 삽입
if (parentNode == t->nil)
{
t->root = newNode;
}
else if (newNode->key < parentNode->key)
{
parentNode->left = newNode;
}
else
{
parentNode->right = newNode;
}
// 삽입된 노드의 색상을 빨간색으로 설정
newNode->color = RBTREE_RED;
newNode->left = t->nil;
newNode->right = t->nil;
// Red-Black 트리의 속성을 유지하기 위해 삽입 후 조정 작업 필요
rbtree_insert_fixup(t, newNode);
return t->root;
}
노드 탐색
// 트리에서 주어진 키를 가진 노드를 찾는 함수
// TODO: 찾기 구현
node_t *rbtree_find(const rbtree *t, const key_t key) // t : 트리, key : 검색 노드 키
{
node_t *node = NULL; // 검색할 노드를 저장할 변수
node = t->root; // 루트 노드부터 검색 시작
// 1. 루트 노드부터 시작하여 키 비교를 통해 왼쪽 또는 오른쪽 자식으로 이동
while (node != t->nil && node->key != key)
{
if (key < node->key)
{
node = node->left; // 키가 현재 노드의 키보다 작으면 왼쪽 자식으로 이동
}
else
{
node = node->right; // 키가 현재 노드의 키보다 크면 오른쪽 자식으로 이동
}
}
// 2. 일치하는 키를 찾으면 해당 노드 반환, 찾지 못하면 NULL 반환
return (node== t->nil) ? NULL : node;
}
가장 작은 키를 가진 노드 찾기
node_t *rbtree_min(const rbtree *t)
{
// 1. 루트 노드부터 시작하여 가장 왼쪽 노드까지 이동
node_t *minNode = t->root;
if (minNode == t->nil)
{
return minNode;
}
while (minNode->left != t->nil)
{
minNode = minNode->left;
}
// 2. 가장 왼쪽 노드 반환
return minNode;
}
가장 큰 키를 가진 노드 찾기
node_t *rbtree_max(const rbtree *t)
{
// 1. 루트 노드부터 시작하여 가장 오른쪽 노드까지 이동
node_t *maxNode = t->root;
if (maxNode == t->nil)
{
return maxNode;
}
while (maxNode->right != t->nil)
{
maxNode = maxNode->right;
}
// 2. 가장 오른쪽 노드 반환
return maxNode;
}
특정 서브 노드에서 가장 작은 값을 찾는 함수(successor)
node_t *rbtree_minimum(rbtree *t, node_t *y)
{
while (y->left != t->nil) // y의 왼쪽 자식이 nil이 되기 전까지 반복
{
y = y->left; // y의 왼쪽 자식값을 y에 담는다
}
return y; // r 반환
}
노드 삭제 후, 삭제된 노드의 자식 노드들을 다른 노드에 연결하는 함수
void rbtree_transplant(rbtree *t, node_t *u, node_t *v)
{
if (u->parent == t->nil) // 삭제된 노드의 부모 노드가 nil이라면(트리의 루트 노트인지 확인)
{
t->root = v; // 루트 노드를 v로 설정(삭제된 노드의 자식 노드 중 하나)
}
else if (u == u->parent->left) // 루트노드가 아니라면 삭제 노드가 부모노드의 왼쪽 자식인지 확인
{
u->parent->left = v; // 왼쪽 자식을 v로 설정
}
else
{ // 그것도 아니라면
u->parent->right = v; // 오른쪽 자식을 v로 설정
}
v->parent = u->parent; // v의 부모를 u의 부모로 설정(v가 u의 위치를 대체)
}
노드 삭제 후 트리 균형을 위한 수정 작업
void rbtree_erase_fixup(rbtree *t, node_t *x) // t : 수정 작업할 트리, x : 삭제된 노드
{
node_t *w;
while (x != t->root && x->color == RBTREE_BLACK)
{
if (x == x->parent->left)
{
w = x->parent->right; // x의 형제 노드 w를 x의 오른쪽 형제 노드로 설정
// case 1:
if (w->color == RBTREE_RED)
{
w->color = RBTREE_BLACK; // w의 색상을 검은색으로 변경
x->parent->color = RBTREE_RED; // x의 부모 노드의 색상을 빨간색으로 변경
rbtree_left_rotate(t, x->parent); // x의 부모 노드를 왼쪽으로 회전
w = x->parent->right; // w를 다시 설정
}
// case 2:
if (w->left->color == RBTREE_BLACK && w->right->color == RBTREE_BLACK)
{
w->color = RBTREE_RED; // w의 색상을 빨간색으로 변경
x = x->parent; // x를 한 단계 위로 이동
}
else
{
// case 3:
if (w->right->color == RBTREE_BLACK)
{
w->left->color = RBTREE_BLACK; // w의 왼쪽 자식 노드의 색상을 검은색으로 변경
w->color = RBTREE_RED; // w의 색상을 빨간색으로 변경
rbtree_right_rotate(t, w); // w를 오른쪽으로 회전
w = x->parent->right; // w를 다시 설정
}
// case 4:
w->color = x->parent->color; // w의 색상을 x의 부모 노드의 색상으로 변경
x->parent->color = RBTREE_BLACK; // x의 부모 노드의 색상을 검은색으로 변경
w->right->color = RBTREE_BLACK; // w의 오른쪽 자식 노드의 색상을 검은색으로 변경
rbtree_left_rotate(t, x->parent); // x의 부모 노드를 왼쪽으로 회전
x = t->root; // x를 루트 노드로 설정
}
}
else
{
// 위의 코드와 동일한 방식으로 x가 x의 부모 노드의 오른쪽 자식인 경우를 처리합니다.
w = x->parent->left;
// case 1:
if (w->color == RBTREE_RED)
{
w->color = RBTREE_BLACK;
x->parent->color = RBTREE_RED;
rbtree_right_rotate(t, x->parent);
w = x->parent->left;
}
// case 2:
if (w->right->color == RBTREE_BLACK && w->left->color == RBTREE_BLACK)
{
w->color = RBTREE_RED;
x = x->parent;
}
else
{
// case 3:
if (w->left->color == RBTREE_BLACK)
{
w->right->color = RBTREE_BLACK;
w->color = RBTREE_RED;
rbtree_left_rotate(t, w);
w = x->parent->left;
}
// case 4:
w->color = x->parent->color;
x->parent->color = RBTREE_BLACK;
w->left->color = RBTREE_BLACK;
rbtree_right_rotate(t, x->parent);
x = t->root;
}
}
}
x->color = RBTREE_BLACK; // 삭제된 노드 x의 색상을 검은색으로 변경
}
노드 삭제
int rbtree_erase(rbtree *t, node_t *p) // t : 삭제 작업 트리, p : 삭제할 노드
{
node_t *y = p; // 삭제할 노드를 y로 설정
color_t y_original_color = y->color; // y의 원래 색상을 저장
node_t *x; // 삭제 후 대체할 노드를 저장할 변수
if (p->left == t->nil)
{
x = p->right; // 삭제할 노드의 오른쪽 자식을 x로 설정
rbtree_transplant(t, p, p->right); // p를 p의 오른쪽 자식으로 대체
}
else if (p->right == t->nil)
{
x = p->left; // 삭제할 노드의 왼쪽 자식을 x로 설정
rbtree_transplant(t, p, p->left); // p를 p의 왼쪽 자식으로 대체
}
else
{
y = rbtree_minimum(t, p->right); // 삭제할 노드의 오른쪽 서브트리에서 가장 작은 노드를 y로 설정
y_original_color = y->color; // y의 원래 색상을 저장
x = y->right; // y의 오른쪽 자식을 x로 설정
if (y->parent == p)
{
x->parent = y; // x의 부모를 y로 설정
}
else
{
rbtree_transplant(t, y, y->right); // y를 y의 오른쪽 자식으로 대체
y->right = p->right; // y의 오른쪽 자식을 p의 오른쪽 자식으로 설정
y->right->parent = y; // y의 오른쪽 자식의 부모를 y로 설정
}
rbtree_transplant(t, p, y); // p를 y로 대체
y->left = p->left; // y의 왼쪽 자식을 p의 왼쪽 자식으로 설정
y->left->parent = y; // y의 왼쪽 자식의 부모를 y로 설정
y->color = p->color; // y의 색상을 p의 색상으로 설정
}
if (y_original_color == RBTREE_BLACK)
{
rbtree_erase_fixup(t, x); // 레드-블랙 트리의 균형을 유지하기 위해 수정 작업을 수행
}
free(p); // 삭제된 노드 p를 메모리에서 해제
return 0; // 삭제 작업 완료
}
중위 순회를 하며 키 값을 배열에 저장하는 함수
void inorder_traversal(node_t *node, int *index, int *arr) {
// 현재 노드가 Nil이거나 인덱스가 유효하지 않으면 순회 중단
if (node == NULL || node->key == 0 || *index <= -1)
{
return;
}
// 왼쪽 서브트리를 순회
inorder_traversal(node->left, index, arr);
// 현재 노드 방문
// 현재 노드의 키를 배열의 현재 index 위치에 저장하고, index 증가
if (*index >= 0)
{
arr[*index] = node->key;
(*index)++;
}
// 오른쪽 서브트리를 순회
inorder_traversal(node->right, index, arr);
}
rb트리의 모든 키를 배열로 변환하는 함수
int rbtree_to_array(const rbtree *t, key_t *arr, const size_t n) {
// 배열 포인터가 유효하지 않거나 배열의 크기가 0이면 오류 코드를 반환
if (arr == NULL || n == 0) {
return -1;
}
int index = 0;
// 트리의 루트부터 시작하여 중위 순회 수행
// 이 순회는 트리의 키를 정렬된 순서로 배열에 채움
inorder_traversal(t->root, &index, arr);
// 순회가 끝난 후, 인덱스는 배열에 저장된 키의 수와 같아야 함
// 인덱스가 배열의 크기를 초과하면 오류 코드를 반환
if (index > n) {
return -1;
}
// 함수 성공 반환
return 0;
}
학습 시간 : 5 ~ 24시
728x90
'크래프톤 정글 > TIL' 카테고리의 다른 글
크래프톤 정글 week05, day33 - 포인터 문제 -1 (2) | 2024.02.21 |
---|---|
크래프톤 정글 week05, day32 - User mode & Kernel mode, 시스템 콜(System Call) (1) | 2024.02.21 |
크래프톤 정글 week04, day30 - rb트리 구현 중 (1) | 2024.02.21 |
크래프톤 정글 week04, day29 - 프로세스, 쓰레드 (1) | 2024.02.20 |
크래프톤 정글 week04, day28 - 배열과 포인터의 관계, 균형 이진 탐색 트리 (1) | 2024.02.20 |