https://blog.csdn.net/dcrmg/article/details/83866368
C 数据结构之十大排序 三大查找
https://blog.csdn.net/pilipilipan/article/details/79702298
十大排序算法代码总览(C++)
二. 分块查找(索引查找)
分块查找是二分查找和顺序查找的一种改进方法,分块查找要求把一个大的线性表分解成若干块,每块中的节点可以任意存放,但块与块之间必须排序。
分块查找要建立一个索引表(占用额外内存空间),把每块中的最大关键码值作为索引表的关键码值,按块的顺序存放到一个辅助数组中,显然这个辅助数组是按关键码值费递减排序的。
查找时,首先在索引表中进行查找,确定要找的节点所在的块。由于索引表是排序的,因此,对索引表的查找可以采用顺序查找或折半查找;然后,在相应的块中采用顺序查找,即可找到对应的节点。
分块查找查找流程:
- 1. 建立索引表(顺序排列的);
- 2. 根据索引表查找到key所在的块;
- 3. 在快内采用顺序查找方法找到对应节点;
操作步骤
step1 先选取各块中的最大关键字构成一个索引表; [1]
step2 查找分两个部分:先对索引表进行二分查找或
顺序查找,以确定待查记录在哪一块中;
然后,在已确定的块中用顺序法进行查找。
#include <iostream>
#include <cmath>
using namespace std;
//定义索引表
typedef struct{
int key;
int address;
}IdxTable;
//节点信息表
typedef struct{
int key;
}NodeTable;
//利用二分查找,定位块号
/*
*idx为索引表
*m为块数
*key为需要查找的关键字
*函数返回关键字所在的块号
*/
int BiSearch(IdxTable idx[],int m,int key){
int low=0,high=m-1,mid;
int found=0;
while(low<=high&&found!=1){
mid=(low+high)/2;
if(key==idx[mid].key){
high=mid-1;
found=1;
}else if(key<idx[mid].key){
high=mid-1;
}else{
low=mid+1;
}
}
//查找完毕,high为key所在的前一个块的块号
return high+1;
}
/*
*node为数据表
*n为数据表的长度
*idx为索引表
*m为索引表的长度,即块数
*s为块长
*key为待查询的关键字
*/
int BlkSearch(NodeTable node[],int n,IdxTable idx[],int m,int s,int key){
int i=BiSearch(idx,m,key);//key所在的块
int begin=idx[i].address;//key所在的块的起始位置
int end;
if(i==m-1){//如果在最后一个块中,上界的位置就是n
end=n-1;
}else {
end=idx[i+1].address;//如果要查询的不在最后一块,那么上界就是下一块的起始位置,即索引表中下一块的地址
}
for(int i=begin;i<end;i++){//在块内顺序查找,找到返回位置,找不到则返回-1
if(node[i].key==key){
return i;
}
}
return -1;
}
int main(){
int a[18]={22,12,13,8,9,20,33,42,44,38,24,48,60,58,74,49,86,53};
NodeTable node[18];
for(int i=0;i<18;i++){
node[i].key=a[i];//初始化节点信息表
}
int b=ceil(18.0/6.0);//总个数除以块长 ,向上取整
//建立索引表
IdxTable idx [3];
idx[0].key=22;idx[0].address=0;
idx[1].key=48;idx[1].address=6;
idx[2].key=86;idx[2].address=12;
int key=48;
int index=BlkSearch(node,18,idx,3,6,key);
if(index==-1){
cout<<"表中不存在"<<key<<"!"<<endl;
}else{
cout<<key<<"在表中的位置是:"<<index<<endl;
}
return 0;
}