B 树C++实现

1.  实现过程参考<算法导论> 第18章B树

/*
 * b_tree.h
 *
 *  Created on: 2011-11-10
 *      Author: guopeng
 */


#ifndef B_TREE_H_
#define B_TREE_H_


#include <stdio.h>
#include <stdlib.h>
#include <algorithm>
#include "circle_queue.h"


static const int B_LIMIT = 3;
static const int MAX_NUM_KEY = 2 * B_LIMIT - 1;
static const int SUB_NUM_KEY = B_LIMIT - 1; //分裂后新节点的孩子数


/***
 * B 树关键性质
 * 1.每个叶节点具有相同的高度(!!节点分裂,中间值会提升到父节点.可称为向上分裂)
 * 2.非根节点至少有t-1个关键字,非根内节点至少有t个子女
 */
struct B_Node {
    int num_key;
    char key[2*B_LIMIT - 1 + 1]; //key[0] no use
    bool leaf_flag;
    struct B_Node * child[2*B_LIMIT + 1]; //child[0] no use
};


struct Root {
int num_node;
struct B_Node *first;
};


// 申请一个节点
struct B_Node &AllocateNode() {
struct B_Node *node = (struct B_Node *)malloc(sizeof(struct B_Node));
node->num_key = 0;
for (int i = 0; i <= MAX_NUM_KEY; i++) {
node->key[i+1] = 'a';
}
node->leaf_flag = false;
for (int i = 0; i < 2 * B_LIMIT; ++i) {
node->child[i + 1] = NULL;
}


return *node;
}


void BTreeCreat(struct Root * &root) {
struct B_Node &node = AllocateNode();
node.leaf_flag = true;
node.num_key = 0;
root->first = &node;
}


// 节点y满的时候(2t-1)个key值时,在插入节点,分裂为两个各含t-1个key值的节点,中间key值被提升到y的双亲节点
// !!!满节点的分裂动作会沿着树向上传播
void BTreeSplitChild(struct B_Node *parent, int pos, struct B_Node *child) {
struct B_Node &new_child = AllocateNode();
new_child.leaf_flag = child->leaf_flag;
new_child.num_key = SUB_NUM_KEY;
for (int j = 1; j <= SUB_NUM_KEY; j++) { //拷贝key值
new_child.key[j] = child->key[j+B_LIMIT];
}


if (!child->leaf_flag) {
for (int j = 1; j <= B_LIMIT; j++) { //拷贝孩子
new_child.child[j] = child->child[j+B_LIMIT];
}
}


child->num_key = SUB_NUM_KEY;


for (int j = parent->num_key + 1; j >= pos+1; j--) { //parent后移一位
parent->child[j+1] = parent->child[j];
}
parent->child[pos+1] = &new_child;
for (int j = parent->num_key; j >= pos; j--) {
parent->key[j+1] = parent->key[j];
}
parent->key[pos] = child->key[B_LIMIT];
parent->num_key += 1;


}


//
void BTreeInsertNonFull(struct B_Node *node, char k) {
int pos = node->num_key;
if (node->leaf_flag) {
while (pos >= 1 && k < node->key[pos]) {
node->key[pos+1] = node->key[pos];
pos--;
}
node->key[pos+1] = k;
node->num_key += 1;
} else {
while (pos >= 1 && k < node->key[pos]) {
pos--;
}
pos += 1;
if (node->child[pos]->num_key == MAX_NUM_KEY) {
BTreeSplitChild(node, pos, node->child[pos]);
if (k > node->key[pos]) {
pos += 1;
}
}
BTreeInsertNonFull(node->child[pos], k);
}
}


void BTreeInsert(struct Root * root, char k) {
struct B_Node *first = root->first;
if (first->num_key == MAX_NUM_KEY) {
struct B_Node &new_node = AllocateNode();
new_node.leaf_flag = false;
new_node.num_key = 0;
new_node.child[1] = first;
root->first = &new_node;


BTreeSplitChild(&new_node, 1, first);
BTreeInsertNonFull(&new_node, k);


} else {
BTreeInsertNonFull(first, k);
}
}


struct B_Node *BTreeSearch(struct B_Node *node, char k, int &index) {
int i = 1;
while (i <= node->num_key && k > node->key[i]) {
i += 1;
}
if (i <= node->num_key && k == node->key[i])
{
index = i;
return node;
}
if (node->leaf_flag) {
index = -1;
return NULL;
} else {
return BTreeSearch(node->child[i], k, index);
}
}
// 保证k的左右子树的个数至少为t-1个
void BTreeDeleteNonSmall(struct B_Node *find_node, int find_index, char k) {
if (find_node->leaf_flag) { //如果是叶子节点,直接删除k
        for (int i = find_index; i < find_node->num_key; i++) {
find_node->key[i] = find_node->key[i+1];
}
find_node->num_key--;
return;
}
if (find_node->child[find_index]->num_key >= B_LIMIT) { //k的前驱子节点, P274-2a
struct B_Node *pre_child = NULL;
printf("in left \n");
do {
pre_child = find_node->child[find_index];
find_node->key[find_index] = pre_child->key[pre_child->num_key];


find_node = pre_child;
find_index = find_node->num_key +1; //最右孩子
} while (!pre_child->leaf_flag);
pre_child->num_key--;
} else if (find_node->child[find_index+1]->num_key >= B_LIMIT) { //k的后继子节点, P274-2b
struct B_Node *pos_child = NULL;
printf("in right \n");
do {
pos_child = find_node->child[find_index+1];
find_node->key[find_index] = pos_child->key[1];
find_node = pos_child;
find_index = 0; //最左孩子
} while (!pos_child->leaf_flag);


for (int i = 2; i <= pos_child->num_key; i++) {
pos_child->key[i-1] = pos_child->key[i];
}
pos_child->num_key--;
}


}


/****
 * 算法导论第18章
 * google过,一致的做法是:
 *  第一步骤:在树中查找被删关键字K所在的地点
 *  第二步骤:进行删去K的操作 ,
 *  BTreeDelete函数遵循上面流程
 */
void BTreeDelete(struct Root * root, char k) {
struct B_Node *find_node = NULL;
int find_index;
find_node = BTreeSearch(root->first, k, find_index);


if (NULL != find_node) {
if (find_node->leaf_flag) { //如果是叶子节点,直接删除k
for (int i = find_index; i < find_node->num_key; i++) {
find_node->key[i] = find_node->key[i+1];
}
find_node->num_key--;
} else {
if ((find_node->child[find_index]->num_key >= B_LIMIT) || (find_node->child[find_index+1]->num_key >= B_LIMIT)) {
BTreeDeleteNonSmall(find_node, find_index, k);
} else { // 两个相邻孩子个数都只有t-1个,合并!!p274-2c
struct B_Node *pre_child = find_node->child[find_index];
struct B_Node *pos_child = find_node->child[find_index+1];
pre_child->num_key += 1;
pre_child->key[pre_child->num_key] = k;// 存在key值
int k_index = pre_child->num_key;
for (int i = find_index; i < find_node->num_key; i++) {
find_node->key[i] = find_node->key[i+1];
find_node->child[i+1] = find_node->child[i+2];
}
find_node->num_key--;


for (int i = 0; i < pos_child->num_key; i++) { // 合并右孩子key值和child值
pre_child->num_key += 1;
pre_child->key[pre_child->num_key] = pos_child->key[i+1];
pre_child->child[pre_child->num_key] = pos_child->child[i+1];
}


delete pos_child;


BTreeDeleteNonSmall(pre_child, k_index, k);
}
}
} else {
printf("not found %c", k);
}




}


// 分层打印, 借助于队列
void BTreePrint(struct Root *root) {


CircleQueue<B_Node> b_queue;
b_queue.Init(100);


struct B_Node *first = root->first;


    b_queue.Push(*first);


    printf("size = %d\n", b_queue.GetSize());


    int base_level = 0;
    struct B_Node *pop_node = (struct B_Node *)malloc(sizeof( struct B_Node));


    printf("\n------------level %d----------------\n", base_level);
    while (b_queue.GetSize() > 0) {


    b_queue.Pop(*pop_node);
    for (int i = 0; i < pop_node->num_key; i++) {
    printf("%c, ", pop_node->key[i+1]);
    }
    printf("\n");


if (!pop_node->leaf_flag) {
base_level++;
printf("\n------------level %d----------------\n", base_level);
for (int i = 0; i < pop_node->num_key+1; i++) { // child数比num_key多1
b_queue.Push(*pop_node->child[i+1]);
}
}
    }


}


#endif /* B_TREE_H_ */

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值