1. 查找的基本概念
data:image/s3,"s3://crabby-images/e44c5/e44c506d9d7b8e1216c0c0ec3931c790d81b83b1" alt=""
2. 静态查找
2.1 顺序查找
c
typedef int KeyType;
typedef int InfoType;
typedef struct
{
KeyType key;
InfoType otherdata;
}SeqList; // 顺序表类型
// 顺序查找
c
int SeqSearch(SeqList R[], int n, int k)
{
int i = n;
R[0].key = k; // R[0].key为查找不成功的监视哨
while (R[i].key != k)
i--;
return i; // 查找成功返回所找元素的索引,否则返回0;
}
2.2 有序表的查找
二分查找
c
int BinarySearch(SeqList R[], int n, int k)
{
int left = 0, right = n - 1;
int mid = 0;
while (left <= right)
{
mid = (left + right) / 2;
if (R[mid].key > k)
right = mid - 1;
else if (R[mid].key < k)
left = mid + 1;
else
return mid;
}
return 0;// 没有找到
}
分块查找(索引顺序查找)
data:image/s3,"s3://crabby-images/08851/0885145cb47f85961b2a38414def682863275f40" alt=""
3. 树表形式的动态查找表
data:image/s3,"s3://crabby-images/6b414/6b4143d4770c915256c790ae83790dd53292c6a0" alt=""
3.1 二叉排序树
data:image/s3,"s3://crabby-images/7859f/7859f60931e17a8d695da54b08e852dacbf4b422" alt=""
二叉排序树的查找操作
c
// 二叉排序树的查找操作
BSTree* BSTSearch(BSTree* t, int k)
{
while (t != NULL)
{
if (t->key > k)
t = t->lchild;
else if (t->key < k)
t = t->rchild;
else
return t;
}
return NULL;
}
二叉排序树的插入操作和二叉树排序树的构造
- 愚蠢的bug,直接拿着main函数传入的指针遍历二叉排序树,导致每次插入节点时都会丢失二叉排序树的根
c
void BSTCreate(BSTree** t, int k)
{
BSTree* pre = NULL;
while ((*t) != NULL)
{
if ((*t)->key > k) {
pre = *t;
*t = (*t)->lchild;
}
else if ((*t)->key < k) {
printf("______________,右子树\n");
pre = *t;
*t = (*t)->rchild;
}
else // 所查节点已经存在
break;
}
//当所查节点不存在时
if (*t == NULL)
{
BSTree* tmp = (BSTree*)malloc(sizeof(BSTree));
tmp->lchild = NULL;
tmp->rchild = NULL;
tmp->key = k;
if (pre != NULL) {
if (pre->key > k) { // 应该插入pre的左孩子
pre->lchild = tmp;
}
else { // 应该插入pre的右孩子
printf("应该插入pre的右孩子\n");
pre->rchild = tmp;
}
}
else { // 二叉排序树还未建立
printf("建立二叉排序树\n");
*t = tmp;
}
}
}
- 正确的方式
c
void BSTCreate(BSTree** t, int k)
{
BSTree* pre = NULL,*current = *t;
while (current != NULL)
{
if (current->key > k) {
pre = current;
current = current->lchild;
}
else if (current->key < k) {
/* printf("______________,右子树\n");*/
pre = current;
current = current->rchild;
}
else // 所查节点已经存在
return;
}
BSTree* tmp = (BSTree*)malloc(sizeof(BSTree));
tmp->lchild = NULL;
tmp->rchild = NULL;
tmp->key = k;
if (pre != NULL) {
if (pre->key > k) { // 应该插入pre的左孩子
pre->lchild = tmp;
}
else { // 应该插入pre的右孩子
//printf("应该插入pre的右孩子\n");
pre->rchild = tmp;
}
}
// 二叉排序树还未建立
else {
printf("建立二叉排序树\n");
*t = tmp;
}
}
删除二叉排序树中的节点
c
void BSTDeleteLeafChild(BSTree* pre, BSTree* current)
{
if (pre->lchild == current) // 待删除节点为pre的左孩子
{
pre->lchild = NULL;
}
else if (pre->rchild == current) {
pre->rchild = NULL;
}
free(current);
current = NULL;
}
void BSTDeleteRightChild(BSTree* pre, BSTree* current)
{
// 待删除节点current只有右孩子,直接将该有孩子替换到待删除节点位置即可
if (pre->lchild == current) // 待删除节点为pre的左孩子
{
pre->lchild = current->rchild;
free(current);
current = NULL;
}
else if (pre->rchild == current) {
pre->rchild = current->rchild;
free(current);
current = NULL;
}
else {
printf("BSTDeleteRightChild:pre和current没有父子关系!!!\n");
}
}
void BSTDeleteLeftChild(BSTree* pre, BSTree* current)
{
// 待删除节点current只有左孩子,直接将该左孩子替换到待删除节点位置即可
if (pre->lchild == current) // 待删除节点为pre的左孩子
{
pre->lchild = current->lchild;
free(current);
current = NULL;
}
else if (pre->rchild == current) {
pre->rchild = current->lchild;
free(current);
current = NULL;
}
else {
printf("BSTDeleteRightChild:pre和current没有父子关系!!!\n");
}
}
// 在二叉排序树种删除某个节点
void BSTDelete(BSTree** t, int k)
{
/*
会出现四种情况
1. 待删除的节点为叶子
2. 待删除的节点只有左孩子
3. 待删除节点只有右孩子
4. 待删除节点左右孩子都有
*/
BSTree* pre = NULL, * current = *t;
while (current != NULL)
{
if (current->key > k) {
pre = current;
current = current->lchild;
}
else if (current->key < k) {
pre = current;
current = current->rchild;
}
else // 节点找到
break;
}
if (current == NULL)
{
printf("该节点没有找到\n");
return;
}
//1. 待删除的节点为叶子
if (current->lchild == NULL && current->rchild == NULL)
{
BSTDeleteLeafChild(pre, current);
}
//2. 待删除的节点只有左孩子
else if (current->lchild != NULL && current->rchild == NULL)
{
BSTDeleteLeftChild(pre, current);
}
//3. 待删除节点只有右孩子
else if (current->lchild == NULL && current->rchild != NULL)
{
BSTDeleteRightChild(pre,current);
}
//4. 待删除节点左右孩子都有
else
{
// 1. 首先找到以待删除节点为根的最左节点
BSTree* t1 = current,*t2 = current;
while (t2->lchild != NULL)
{
t1 = t2;
t2 = t2->lchild;
}
current->key = t2->key;
2. 删除最左节点
if(t2->rchild!=NULL)
BSTDeleteRightChild(t1, t2);
else {
t1->lchild = NULL;
free(t2);
t2 = NULL;
}
}
}
3.2 平衡二叉树(AVL)
data:image/s3,"s3://crabby-images/29379/29379e5c4df2f0a96ac93f565f8c6fdf4f31c903" alt=""
data:image/s3,"s3://crabby-images/813da/813da0032902f3a727dc30a8023b8dbed22d0bdc" alt=""
红黑树
3.3 B树和B+树
B树
B树的插入
- 例子
data:image/s3,"s3://crabby-images/54b3d/54b3dc830abff16818a92bdfce267276b4140108" alt=""
- 例子
- 插入15
- 插入35
- 插入95
B树的删除
data:image/s3,"s3://crabby-images/6416d/6416de0a962eaf474008731ef9363c92aa46db72" alt=""
-
例子
-
删除92
-
删除80
data:image/s3,"s3://crabby-images/3aa22/3aa227f6e754117908b2f8174f0dd3b8cf84f515" alt=""
data:image/s3,"s3://crabby-images/aacb4/aacb4547e79f9ae932f2b9ad9ffa23f9c6640c94" alt=""
-
- 删除70
- 删除70
data:image/s3,"s3://crabby-images/73cd8/73cd85e827a3fade546ee06520215638bf50e167" alt=""
B+树
两者的区别
4. 哈希
4.1 哈希表与哈希方法
data:image/s3,"s3://crabby-images/09459/09459a96ec411e96bc4381e9fc9403be46d87f28" alt=""
4.2 哈希函数
直接定址法
data:image/s3,"s3://crabby-images/94ead/94ead032fcd8a2b4fd8394525df0db43e8f24f22" alt=""
除留余数法
data:image/s3,"s3://crabby-images/f5fbc/f5fbc6e66a96578dacf5a32d23d780d57b7847a7" alt=""
数字分析法
data:image/s3,"s3://crabby-images/0fa50/0fa500cf33db17a6db40305e829b87a567b99a1c" alt=""
平方取中法
data:image/s3,"s3://crabby-images/e4127/e4127f006ef6b92b98641179500c0521d264ec7c" alt=""
折叠法
data:image/s3,"s3://crabby-images/ee1c3/ee1c325bae48fcff6103981ea8c7fd334efd6dc3" alt=""
4.3 处理冲突的方法
data:image/s3,"s3://crabby-images/1d492/1d49238a8a5f07d666a39badc14e078455819f56" alt=""
data:image/s3,"s3://crabby-images/083a8/083a8d384ff9753a430a3a05fe1a20c831313c8e" alt=""
闭散列表
开放地址法
data:image/s3,"s3://crabby-images/99a7b/99a7b1a363e96f1263a709f10cf8783d5eedcc0f" alt=""
再散列法
data:image/s3,"s3://crabby-images/5a50b/5a50bdf8689418ad0da8348d860387fe10847859" alt=""
开散列表
data:image/s3,"s3://crabby-images/62a6d/62a6da7a5435bdeb087e1c0cfd3a8a31dcc96c27" alt=""