分块查找代码实现

1.方法描述(摘自百度百科……懒)

            分块查找要求把一个大的线性表分解成若干块,每块中的节点可以任意存放,但块与块之间必须排序。假设是按关键码值非递减的,那么这种块与块之间必须满足已排序要求,实际上就是对于任意的i,第i块中的所有节点的关键码值都必须小于第i+1块中的所有节点的关键码值。此外,还要建立一个索引表,把每块中的最大关键码值作为索引表的关键码值,按块的顺序存放到一个辅助数组中,显然这个辅助数组是按关键码值费递减排序的。查找时,首先在索引表中进行查找,确定要找的节点所在的块。由于索引表是排序的,因此,对索引表的查找可以采用顺序查找或折半查找;然后,在相应的块中采用顺序查找,即可找到对应的节点。

           分块查找在现实生活中也很常用。例如,一个学校有很多个班级,每个班级有几十个学生。给定一个学生的学号,要求查找这个学生的相关资料。显然,每个班级的学生档案是分开存放的,没有任何两个班级的学生的学号是交叉重叠的,那么最好的查找方法实现确定这个学生所在的班级,然后再在这个学生所在班级的学生档案中查找这个学生的资料。上述查找学生资料的过程,实际上就是一个典型的分块查找。

2.操作步骤

step1 先选取各块中的最大关键字构成一个索引表; [1] 

step2 查找分两个部分:先对索引表进行二分查找或

顺序查找,以确定待查记录在哪一块中;

然后,在已确定的块中用顺序法进行查找。

3.代码实现

#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;
} 

 以上代码以该数组为例:

2212138920334244382448605874498653
01234567891011121314151617
  • 1
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值