- 使用 Spring Initializr 创建 Spring Boot 应用程序
- 在Spring Boot中配置Cassandra
- 在 Spring Boot 上配置 Tomcat 连接池
- 将Camel消息路由到嵌入WildFly的Artemis上
https://www.cs.usfca.edu/~galles/visualization/BTree.html
为啥要有b树,我们想想,我们之前学过或者使用过的二叉树红黑树…这些数据结构,发现一个问题就是在数据量很大的时候那么我们树的高度是非常高的,在我们日常生活中存储数据最常见的外存就是磁盘。
而磁盘是快设备,也就是说磁盘的读写单位是以块为单位,一般地块大小从0.5k到4k。
即使你只读取一个字节,磁盘也是将包含该字节的所有数据读取到硬盘中。而在磁盘读取过程中,最占用时间的是磁盘的寻道,也就是磁头在盘片上找到需要读取的块所在位置的时间,而在盘片上顺序读取数据的所花的时间是占比比较小的。而要减少外存上花的时间,就可以从减少读盘次数以及减少寻道时间着手,B树采取的方法就是,就充分的利用盘块的空间,在一个盘块中尽可能多的存储信息,或者在连续的盘块地址上存储尽可能多的信息。在数据结构上的变化就是每个节点存储多个key信息以及包含多个子节点。增加节点的分支树,就可以使得这棵树的高度降低
假设: 是5阶B树那么高度为3的B树可以存3^5-1=242
如果是11阶呢? 3^11-1=177146
以此类推…
那么对比二叉树呢? 我们拿满二叉树来举例,如果高度为3的满二叉树那么能存多少值呢?:2^3-1=7
总结: 在同样高度的情况下B树,是比二叉树存储数据多得多,这样就加快了磁盘寻道的速度了.
这里解释一下B-树
就是B树
(x是节点, x.key表示节点的值 ,n表示阶或者长度)
x.key1 <= x.key2<= x.key3 <= .... <= x.keyn。
也就是x中的key以有序排列的this.degree = degree;
//计算分割下标
this.ceil = (int) Math.ceil((double) degree / 2) - 1;
//然后根据这个位置的值,小的放左,大的放右......具体细节下面会讲
n> x>=2
2为最小,最大不能超过n阶反正一句话,无论什么时候插入都是在叶子节点
这个比插入还麻烦,但是只要把条件写好也能行,很多人都不愿意写删除操作,而使用删除标记代替也行,看个人
必须遵守的规则: 保证删除叶子节点key个数必须是大于等于Math.ceil(m/2)-1
删除步骤
前驱: 取叶子节点最大值
后驱: 取叶子节点最小值
选择一种就行,我使用的是前驱
反正一句话,无论什么时候删除都是在叶子节点
有人就会说删除后如果叶子节点key的个数小于Math.ceil(m/2)-1,那么不就不满足b树的最小key的个数吗?
我就默默的回答一句话,在叶子节点的话这个细节是可以忽略的,因为下次插入的时候会补上的因为只要满足key=阶级-1才会进行分裂
有人就又问了,能在删除过程解决这个小小的细节吗?
答案: 能,但是太复杂了, 需要设计到节点的重构,每个情况重构方法都不同,最差一直重构到root节点,就拿各种情况而言至少5种以上我就简单举例说明下:
B树的查找和二叉树查找类似,首先在当前节点中查找,如果没有并且存在孩子节点,就递归的到可能存在该key的孩子节点中查找。
不同的是,B树节点有多个key,需要每个都比较,如果key的个数很多的话(key>1000),为了提高性能,可以使用二分法加速节点中的查找。
在查询的基础上,修改内容就行,这是比较简单的
package com.tree.btree;
import com.tree.binarytree.Node;
import org.springframework.util.Assert;
import java.lang.reflect.Field;
import java.util.*;
/**
*
* @author huanmin
*/
public class BTree<T> {
private final int degree;
private BTreeNode<T> root;
private final int ceil; //分割值
//初始化阶级
public BTree(int degree) {
//阶不能小于2
if (degree < 2) {
throw new IllegalArgumentException("degree mustn't < 2");
}
this.degree = degree;
this.ceil = (int) Math.ceil((double) degree / 2) - 1; //计算分割下标
}
class BTreeNode<n> {
private int keySiez;//keys 从1开始计算
private Object[] keys; //存储key (这个key我们可以设计一个key,value形式的对象,利用反射)
private int sonNodeSize;//nodeSize 从1开始计算
private BTreeNode<n>[] sonbTreeNodes; //存储全部子节点
private BTreeNode<n> parent; //存储父节点,这样便于操作
private String sign; //标记,用于特殊方面...
public BTreeNode() {
//设置最大key的长度(degree-1) ,要多预留一个用于交互(degree)
this.keys = new Object[degree];
//设置子节点最大长度(degree) ,要多预留一个用于交互(degree+1)
this.sonbTreeNodes = new BTreeNode[degree + 1];
}
private int getKeyLength() {
return keySiez;
}
//判断节点的key是否满了
private boolean getKeyFull() {
return keySiez == degree;
}
private int getSonNodeSize() {
return sonNodeSize;
}
private void setSonbTreeNodes(BTreeNode<n>[] bTreeNode, Integer index) {
if (index == null) {
for (BTreeNode<n> nBTreeNode : bTreeNode) {
sonbTreeNodes[sonNodeSize] = nBTreeNode;
sonNodeSize++;
}
return;
}
for (BTreeNode<n> nBTreeNode : bTreeNode) {
sonbTreeNodes[index] = nBTreeNode;
index++;
sonNodeSize++;
}
}
private void setSonbTreeNode(BTreeNode<n> bTreeNode) {
sonbTreeNodes[sonNodeSize] = bTreeNode;
sonNodeSize++;
}
private synchronized void setKey(n key) {
//如果是空的那么直接插入
if (keys[0] == null) {
keys[0] = key;
keySiez++;
return;
}
//进行从小到大插入值
//1.先获取插入的位置
Integer location = null;
long newKeyId = getClassId(key);
for (int i = 0; i < getKeyLength(); i++) {
long OldKeyId = getClassId(keys[i]);
//拿到插入的下标
if (OldKeyId > newKeyId) {
location = i;
break;
}
}
if (location == null) {
//表示已经查询到最后了也没有比他大的值了那么直接就插入到最后null位置
keys[getKeyLength()] = key;
keySiez++;
return;
}
//从下标以后的值往后移动
moveArrayKey(keys, getKeyLength(), location, 1);
//将值插入到指定位置
keys[location] = key;
//当前值的个数加一
keySiez++;
}
//分裂成2份返回中间值
private Map<String, Object> bTreeNodeSplit(BTreeNode<n> parent) {
Map<String, Object> map = new HashMap<>();
map.put("center", keys[ceil]);
BTreeNode<n> bTreeNode1 = new BTreeNode<n>();
//值的分割,中间值不要
for (int i = 0; i < ceil; i++) {
bTreeNode1.setKey((n) keys[i]);
}
BTreeNode<n> bTreeNode2 = new BTreeNode<n>();
for (int i = ceil + 1; i < getKeyLength(); i++) {
bTreeNode2.setKey((n) keys[i]);
}
//将子节点的引用分配给分割后的节点,并且调整父节点引用
if (getSonNodeSize() > 0) {
//满足分割条件,子节点>0 那么一定是 key+1的个数
for (int i = 0; i <= ceil; i++) {
if (sonbTreeNodes[i] != null) {
bTreeNode1.setSonbTreeNode(sonbTreeNodes[i]);
sonbTreeNodes[i].parent = bTreeNode1;
}
}
for (int i = ceil + 1; i < getSonNodeSize(); i++) {
if (sonbTreeNodes[i] != null) {
bTreeNode2.setSonbTreeNode(sonbTreeNodes[i]);
sonbTreeNodes[i].parent = bTreeNode2;
}
}
}
BTreeNode<n>[] sonbTreeNodes = new BTreeNode[2];
sonbTreeNodes[0] = bTreeNode1;
sonbTreeNodes[1] = bTreeNode2;
map.put("sonbTreeNodes", sonbTreeNodes);
//添加新节点的父节点
bTreeNode1.parent = parent;
bTreeNode2.parent = parent;
//移动节点位置(一般分裂都是2),清除原来旧节点,然后空出需要插入节点的位置,返回新插入起始下标
Integer integer = moveNode(parent, this, 2);
map.put("fromIdnex", integer);
return map;
}
/**
* 删除旧节点,空出需要插入节点的位置
*
* @param parent 父节点
* @param delNode 需要删除的节点
* @param size 需要插入的节点个数
* @return 返回需要插入下标的起始位置
*/
private Integer moveNode(BTreeNode<n> parent, BTreeNode<n> delNode, int size) {
if (parent.getKeyLength() <= 0) {
return null;
}
Integer index = null;
int sonNodeSize = parent.getSonNodeSize();
for (int i = 0; i < sonNodeSize; i++) {
//找到当前节点的下标
if (parent.sonbTreeNodes[i] == delNode) {
index = i;
}
}
//将下标后的所有值往后移动,移动size位
if (index == sonNodeSize - 1) {
//如果是最后一位那么直接置空就行,然后返回起始位置
parent.sonbTreeNodes[parent.sonNodeSize - 1] = null;
} else {
parent.sonbTreeNodes[index] = null;
moveArrayNode(parent.sonbTreeNodes, sonNodeSize, index, size);
}
//返回起始位置
parent.sonNodeSize--;
return index;
}
@Override
public String toString() {
return "BTreeNode{" +
"keySiez=" + keySiez +
", keys=" + Arrays.toString(keys) +
", nodeSize=" + sonNodeSize +
", sonbTreeNodes=" + Arrays.toString(sonbTreeNodes) +
'}';
}
}
//插入逻辑
public void add(T data) throws Exception {
//如果root为空那么创建新的节点
if (root == null) {
BTreeNode<T> newNode = new BTreeNode<T>();
insert(newNode, data);
root = newNode;
return;
}
//root节点key,没有插入满,并且还没有子节点,那么继续插入
if (!root.getKeyFull() && root.getSonNodeSize() == 0) {
insert(root, data);
return;
}
//此刻root的key已经满了那么进行分裂节点
if (root.getSonNodeSize() == 0) {
nodeSplit(root);
}
//如果root节点的子节点不为空那么,一直深入到叶子节点
//找到对应的叶子节点
BTreeNode<T> tbTreeNode = selectLeaf(data);
long newKey = getClassId(data); //拿到插值的索引
int keyLength = tbTreeNode.getKeyLength();
for (int i = 0; i < keyLength; i++) {
long oldKey = getClassId(tbTreeNode.keys[i]);//节点对应的索引
//如果发现当前的值存在了,那么直接提示报错,不然这样会影响树的结构,(如果需要插入重复存在的数据,可以使用链表来关联起来)
if (oldKey == newKey) {
throw new Exception("repetition data , keyID: " + newKey);
}
}
//将值插入找到的叶子节点
insert(tbTreeNode, data);
}
//删除逻辑
public boolean del(T data) {
Map<String, Object> node1 = getNode(data);
if (node1 == null) {
return false;
}
BTreeNode<T> node = (BTreeNode<T>) node1.get("node"); //拿到叶子节点
int delIndex = (int) node1.get("index");
// 到这一步 已经找到需要操作的节点,和要删除这个节点那个值的下标了
//如果删除的值是在叶子节点中
delLeaf(node, delIndex);
//删除的值不是叶子节点,那么需要进行复杂的操作,进行前驱或者后驱到叶子节点取值覆盖删除的key
delPrecursorOrSucceed(node, delIndex);
return true;
}
private void delPrecursorOrSucceed(BTreeNode<T> node, int delIndex) {
if (node.getSonNodeSize() > 0) {
//前驱,delIndex-1 节点下的最大子节点一直到叶子节点,然后取叶子节点的最大值
BTreeNode<T> sonbTreeNode = node.sonbTreeNodes[delIndex];
while (sonbTreeNode.getSonNodeSize() > 0) {
sonbTreeNode = sonbTreeNode.sonbTreeNodes[sonbTreeNode.getSonNodeSize() - 1];
}
//拿到前驱最大叶子节点的最大key
node.keys[delIndex] = sonbTreeNode.keys[sonbTreeNode.getKeyLength() - 1];
delArrayKey(sonbTreeNode, sonbTreeNode.getKeyLength() - 1);
}
}
private void delLeaf(BTreeNode<T> node, int delIndex) {
//如果是叶子节点
if (node.getSonNodeSize() == 0) {
delArrayKey(node, delIndex);
}
}
//数组删除-后覆盖删除位置
private void delArrayKey(BTreeNode<T> node, int delIndex) {
int keyLength1 = node.getKeyLength();
if (keyLength1 < 1) {
node.keys[delIndex] = null;
} else {
//将后面的值覆盖删除的值
for (int i = delIndex; i < keyLength1; i++) {
node.keys[i] = node.keys[i + 1];
}
}
node.keySiez--;
}
/**
* @param array 位移的数组
* @param arrayLength 数组长度
* @param moveindex 位移的位置
* @param size 向后位移几位
*/
private void moveArrayNode(BTreeNode[] array, int arrayLength, int moveindex, int size) {
for (int i = arrayLength - 1; i >= moveindex; i--) {
//把当前值放入,数组后i+size位置
array[i + size - 1] = array[i];
}
}
/**
* @param array 位移的数组
* @param arrayLength 数组长度
* @param moveindex 位移的位置
* @param size 向后位移几位
*/
private void moveArrayKey(Object[] array, int arrayLength, int moveindex, int size) {
//从下标以后的值往后移动(包括下标)
for (int i = arrayLength - 1; i >= moveindex; i--) {
array[i + size] = array[i];
}
}
//查询指定值
public T getData(T data) {
Map<String, Object> node = getNode(data);
if (node==null) {
return null;
}
BTreeNode<T> node1 = (BTreeNode<T>) node.get("node");
Object o = node1.keys[(int) node.get("index")];
return ( T)o;
}
//修改指定值
public boolean getUpdate(T oldData,T newData) {
Map<String, Object> node = getNode(oldData);
if (node==null) {
return false;
}
BTreeNode<T> node1 = (BTreeNode<T>) node.get("node");
node1.keys[(int) node.get("index")]=newData;
return true;
}
private Map<String, Object> getNode(T data) {
BTreeNode<T> node = root; //拿到叶子节点
long newKey = getClassId(data); //拿到索引
while (node.getSonNodeSize() > 0) {
int keyLength = node.getKeyLength();
//找比插入值大的下标,那么就是所对应的子节点下标,如果没找到那么就是最后一个子节点,然后继续深入
Integer max = null;
int state = 0;
for (int i = 0; i < keyLength; i++) {
long oldKey = getClassId(node.keys[i]);//节点对应的索引
//找到值了,返回节点
if (oldKey == newKey) {
state = 1;
break;
}
if (oldKey > newKey) {
max = i;
break;
}
}
//找到了
if (state == 1) {
break;
}
//没找到那么取最后一个子节点
if (max == null) {
max = node.getSonNodeSize() - 1;
}
node = node.sonbTreeNodes[max];
}
//遍历找到的节点,或者叶子节点,找到对应的值的下标
Integer index = null;
int keyLength = node.getKeyLength();
for (int i = 0; i < keyLength; i++) {
long oldKey = getClassId(node.keys[i]);//节点对应的索引
//找到值了,返回节点
if (oldKey == newKey) {
index = i;
break;
}
}
if (index == null) {
return null;
}
Map<String, Object> map = new HashMap<>();
map.put("node", node);
map.put("index", index);
return map;
}
//查询叶子节点逻辑
private BTreeNode<T> selectLeaf(T data) {
BTreeNode<T> node = root; //拿到叶子节点
long newKey = getClassId(data); //拿到索引
while (node.getSonNodeSize() > 0) {
int keyLength = node.getKeyLength();
//找比插入值大的下标,那么就是所对应的子节点下标,如果没找到那么就是最后一个子节点,然后继续深入
Integer max = null;
for (int i = 0; i < keyLength; i++) {
long oldKey = getClassId(node.keys[i]);//节点对应的索引
if (oldKey > newKey) {
max = i;
break;
}
}
//没找到那么取最后一个子节点
if (max == null) {
max = node.getSonNodeSize() - 1;
}
node = node.sonbTreeNodes[max];
}
return node;
}
private void insert(BTreeNode<T> node, T data) {
//先插入值,然后在进行判断key是否满了,如果满了那么进行分割
node.setKey(data);
//插入不进去了满了
if (node.getKeyFull()) {
//进行分割节点
nodeSplit(node);
}
}
//节点分割
private void nodeSplit(BTreeNode<T> bTreeNode) {
//判断是否是root节点
if (bTreeNode == root) {
//如果是root节点那么就创建新节点代替root节点然后将原来的root节点进行分割,之后进行关联父节点
BTreeNode<T> newRootNode = new BTreeNode<T>();
Map<String, Object> map = bTreeNode.bTreeNodeSplit(newRootNode);
//添加节点值
newRootNode.setKey((T) map.get("center"));
//添加分裂后的子节点
newRootNode.setSonbTreeNodes((BTreeNode<T>[]) map.get("sonbTreeNodes"), (Integer) map.get("fromIdnex"));
//将新节点代替原来的root节点
root = newRootNode;
} else {
BTreeNode<T> oldNode = bTreeNode; //获取当前节点
//一直向上查找,直到key不满为止
while (oldNode.getKeyFull()) {
if (oldNode == root) {
//如果是root节点的话,就使用root节点的分割方式
nodeSplit(oldNode);
break;
} else {
//子节点分割
Map<String, Object> map = oldNode.bTreeNodeSplit(oldNode.parent);
//给父节点,添加节点值
oldNode.parent.setKey((T) map.get("center"));
//添加分裂后的子节点,到父节点
oldNode.parent.setSonbTreeNodes((BTreeNode<T>[]) map.get("sonbTreeNodes"), (Integer) map.get("fromIdnex"));
//然后继续向上查找
oldNode = oldNode.parent;
}
}
}
}
//通过反射拿到对象内的id值 ,(不允许字符串类型 ,只能是数值类型,或者对象类型里有id字段)
// long OldKeyId = getClassId(o);
// long newKeyId = getClassId(key);
private long getClassId(Object o) {
if (o instanceof Number) {
Number data = (Number) (o);
return data.longValue();
}
long l = 0L;
try {
Class<?> aClass = o.getClass();
Field field = aClass.getDeclaredField("id");
field.setAccessible(true);
l = (long) field.get(o);
} catch (NoSuchFieldException | IllegalAccessException e) {
e.printStackTrace();
}
return l;
}
public void show() {
//给节点打标
BTreeNode<T> node = root;
while (node.getSonNodeSize() > 0) {
node.sonbTreeNodes[0].sign = "Front";
node = node.sonbTreeNodes[0];
}
node = root;
while (node.getSonNodeSize() > 0) {
node.sonbTreeNodes[node.getSonNodeSize() - 1].sign = "Ent";
node = node.sonbTreeNodes[node.getSonNodeSize() - 1];
}
//变量节点,以树形式展示
Queue<BTreeNode<T>> queue = new LinkedList<BTreeNode<T>>();
queue.offer(root);
while (!queue.isEmpty()) {
BTreeNode<T> tree = queue.poll();
if (tree == root) {
System.out.print("front------------------------------------");
}
//一层一层打印节点
if ("Front".equals(tree.sign)) {
System.out.print("front------------------------------------");
}
System.out.print("|\t");
for (int i = 0; i < tree.getKeyLength(); i++) {
long classId = getClassId(tree.keys[i]);
System.out.print(classId + ",");
}
System.out.print("\t|-");
if ("Ent".equals(tree.sign)) {
System.out.println("------------------------------------Ent");
}
if (tree == root) {
System.out.println("------------------------------------Ent");
}
for (int i = 0; i < tree.getSonNodeSize(); i++) {
queue.offer(tree.sonbTreeNodes[i]);
}
}
}
}
点赞 -收藏-关注-便于以后复习和收到最新内容有其他问题在评论区讨论-或者私信我-收到会在第一时间回复如有侵权,请私信联系我感谢,配合,希望我的努力对你有帮助^_^
关于 B 树与 B+ 树,网上有一个比较经典的问题:为什么 MongoDb 使用 B 树,而 MySQL 索引使用 B+ 树? 但实际上 MongoDb 真的用的是 B 树吗?
如何将 R* Tree 实现为持久(基于磁盘)树?保存 R* 树索引或保存叶值的文件的体系结构是什么? 注意:此外,如何在这种持久性 R* 树中执行插入、更新和删除操作? 注意事项二:我已经实现了一个
目前,我正在努力用 Java 表示我用 SML 编写的 AST 树,这样我就可以随时用 Java 遍历它。 我想知道是否应该在 Java 中创建一个 Node 类,其中包含我想要表示的数据,以及一个数
我之前用过这个库http://www.cs.umd.edu/~mount/ANN/ .但是,它们不提供范围查询实现。我猜是否有一个 C++ 范围查询实现(圆形或矩形),用于查询二维数据。 谢谢。 最佳
在进一步分析为什么MySQL数据库索引选择使用B+树之前,我相信很多小伙伴对数据结构中的树还是有些许模糊的,因此我们由浅入深一步步探讨树的演进过程,在一步步引出B树以及为什么MySQL数据库索引选择
书接上回,今天和大家一起动手来自己实现树。 相信通过前面的章节学习,大家已经明白树是什么了,今天我们主要针对二叉树,分别使用顺序存储和链式存储来实现树。 01、数组实现 我们在上一节中说过,
书节上回,我们接着聊二叉树,N叉树,以及树的存储。 01、满二叉树 如果一个二叉树,除最后一层节点外,每一层的节点数都达到最大值,即每个节点都有两个子节点,同时所有叶子节点都在最后一层,则这个
树是一种非线性数据结构,是以分支关系定义的层次结构,因此形态上和自然界中的倒挂的树很像,而数据结构中树根向上树叶向下。 什么是树? 01、定义 树是由n(n>=0)个元素节点组成的
操作系统的那棵“树” 今天从一颗 开始,我们看看如何从小树苗长成一颗苍天大树。 运转CPU CPU运转起来很简单,就是不断的从内存取值执行。 CPU没有好好运转 IO是个耗费时间的活,如果CPU在取值
我想为海洋生物学类(class)制作一个简单的系统发育树作为教育示例。我有一个具有分类等级的物种列表: Group <- c("Benthos","Benthos","Benthos","Be
我从这段代码中删除节点时遇到问题,如果我插入数字 12 并尝试删除它,它不会删除它,我尝试调试,似乎当它尝试删除时,它出错了树的。但是,如果我尝试删除它已经插入主节点的节点,它将删除它,或者我插入数字
B+ 树的叶节点链接在一起。将 B+ 树的指针结构视为有向图,它不是循环的。但是忽略指针的方向并将其视为链接在一起的无向叶节点会在图中创建循环。 在 Haskell 中,如何将叶子构造为父内部节点的子
我在 GWT 中使用树控件。我有一个自定义小部件,我将其添加为 TreeItem: Tree testTree = new Tree(); testTree.addItem(myWidget); 我想
它有点像混合树/链表结构。这是我定义结构的方式 struct node { nodeP sibling; nodeP child; nodeP parent; char
我编写了使用队列遍历树的代码,但是下面的出队函数生成错误,head = p->next 是否有问题?我不明白为什么这部分是错误的。 void Levelorder(void) { node *tmp,
例如,我想解析以下数组: var array1 = ["a.b.c.d", "a.e.f.g", "a.h", "a.i.j", "a.b.k"] 进入: var json1 = { "nod
问题 -> 给定一棵二叉树和一个和,确定该树是否具有从根到叶的路径,使得沿路径的所有值相加等于给定的和。 我的解决方案 -> public class Solution { public bo
我有一个创建 java 树的任务,它包含三列:运动名称、运动类别中的运动计数和上次更新。类似的东西显示在下面的图像上: 如您所见,有 4 种运动:水上运动、球类运动、跳伞运动和舞蹈运动。当我展开 sk
我想在 H2 数据库中实现 B+ Tree,但我想知道,B+ Tree 功能在 H2 数据库中可用吗? 最佳答案 H2 已经使用了 B+ 树(PageBtree 类)。 关于mysql - H2数据库
假设我们有 5 个字符串数组: String[] array1 = {"hello", "i", "cat"}; String[] array2 = {"hello", "i", "am"}; Str
我是一名优秀的程序员,十分优秀!