数据结构:分块查找

分块查找,也叫索引顺序查找,算法实现除了需要查找表本身之外,还需要根据查找表建立一个索引表。例如图 1,给定一个查找表,其对应的索引表如图所示:

在这里插入图片描述

图 1 查找表及其对应的索引表

1 中,查找表中共 18 个查找关键字,将其平均分为 3 个子表,对每个子表建立一个索引,索引中包含中两部分内容:该子表部分中最大的关键字以及第一个关键字在总表中的位置,即该子表的起始位置。

建立的索引表要求按照关键字进行升序排序,查找表要么整体有序,要么分块有序。

分块有序指的是第二个子表中所有关键字都要大于第一个子表中的最大关键字,第三个子表的所有关键字都要大于第二个子表中的最大关键字,依次类推。

块(子表)中各关键字的具体顺序,根据各自可能会被查找到的概率而定。如果各关键字被查找到的概率是相等的,那么可以随机存放;否则可按照被查找概率进行降序排序,以提高算法运行效率。

分块查找的具体实现

所有前期准备工作完成后,开始在此基础上进行分块查找。分块查找的过程分为两步进行:

  1. 确定要查找的关键字可能存在的具体块(子表);
  2. 在具体的块中进行顺序查找。

以图 1 中的查找表为例,假设要查找关键字 38 的具体位置。首先将 38 依次和索引表中各最大关键字进行比较,因为 22 < 38 < 48,所以可以确定 38 如果存在,肯定在第二个子表中。

由于索引表中显示第二子表的起始位置在查找表的第 7 的位置上,所以从该位置开始进行顺序查找,一直查找到该子表最后一个关键字(一般将查找表进行等分,具体子表个数根据实际情况而定)。结果在第 10 的位置上确定该关键字即为所找。

提示:在第一步确定块(子表)时,由于索引表中按照关键字有序,所有可以采用折半查找算法。而在第二步中,由于各子表中关键字没有严格要求有序,所以只能采用顺序查找的方式。

具体实现代码:

#include <stdio.h>
#include <stdlib.h>

struct index {//定义块的结构int key;
    int start;
} newIndex[3];//定义结构体数组int search(int key, int a[]);

int cmp(const void *a,const void* b){
    return (*(struct index*)a).key>(*(struct index*)b).key?1:-1;
}
int main(){
    int i, j=-1, k, key;
    int a[] = {33,42,44,38,24,48, 22,12,13,8,9,20,  60,58,74,49,86,53};
//确认模块的起始值和最大值for (i=0; i<3; i++) {
        newIndex[i].start = j+1;//确定每个块范围的起始值
        j += 6;
        for (int k=newIndex[i].start; k<=j; k++) {
            if (newIndex[i].key<a[k]) {
                newIndex[i].key=a[k];
            }
        }
    }
//对结构体按照 key 值进行排序
    qsort(newIndex,3, sizeof(newIndex[0]), cmp);

//输入要查询的数,并调用函数进行查找printf("请输入您想要查找的数:\n");
    scanf("%d", &key);
    k = search(key, a);
//输出查找的结果if (k>0) {
        printf("查找成功!您要找的数在数组中的位置是:%d\n",k+1);
    }else{
        printf("查找失败!您要找的数不在数组中。\n");
    }
    return 0;
}
int search(int key, int a[]){
    int i, startValue;
    i = 0;
    while (i<3 && key>newIndex[i].key) {//确定在哪个块中,遍历每个块,确定key在哪个块中
        i++;
    }
    if (i>=3) {//大于分得的块数,则返回0return -1;
    }
    startValue = newIndex[i].start;//startValue等于块范围的起始值while (startValue <= startValue+5 && a[startValue]!=key)
    {
        startValue++;
    }
    if (startValue>startValue+5) {//如果大于块范围的结束值,则说明没有要查找的数return -1;
    }
    return startValue;
}

运行结果:

请输入您想要查找的数:
22
查找成功!您要找的数在数组中的位置是:7

分块查找的性能分析

分块查找算法的运行效率受两部分影响:查找块的操作和块内查找的操作。查找块的操作可以采用顺序查找,也可以采用折半查找(更优);块内查找的操作采用顺序查找的方式。相比于折半查找,分块查找时间效率上更低一些;相比于顺序查找,由于在子表中进行,比较的子表个数会不同程度的减少,所有分块查找算法会更优。总体来说,分块查找算法的效率介于顺序查找和折半查找之间。

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
查找子系统是数据结构中的一个重要部分,它是指在一个数据集合中查找某个特定元素的过程。常见的查找算法有顺序查找、二分查找、分块查找、哈希查找等。下面是这些算法的简单介绍和示例: 1. 顺序查找:将待查找值与查找表中的数据依次进行比较,直到找到为止。 ```python def sequential_search(lst, key): for i in range(len(lst)): if lst[i] == key: return i return -1 ``` 2. 二分查找:对于有序表,每次将待查找区间缩小一半,直到找到为止。 ```python def binary_search(lst, key): low, high = 0, len(lst) - 1 while low <= high: mid = (low + high) // 2 if lst[mid] < key: low = mid + 1 elif lst[mid] > key: high = mid - 1 else: return mid return -1 ``` 3. 分块查找:将查找表分成几块,块与块之间是有序的,形成一张索引顺序表,在索引顺序表中查找待查找值,用折半查找的方法确定待查找值在哪一块,然后在确定的块中用顺序查找的方法进行查找。 ```python def block_search(lst, key, block_size): blocks = [lst[i:i+block_size] for i in range(0, len(lst), block_size)] index = [block[0] for block in blocks] block_id = binary_search(index, key) if block_id == -1: return -1 else: block = blocks[block_id] for i in range(len(block)): if block[i] == key: return block_id * block_size + i return -1 ``` 4. 哈希查找:基于哈希函数的查找方法,将待查找值通过哈希函数映射到一个桶中,然后在桶中进行查找。 ```python class HashTable: def __init__(self, size): self.size = size self.table = [[] for _ in range(size)] def hash(self, key): return key % self.size def insert(self, key, value): h = self.hash(key) self.table[h].append((key, value)) def search(self, key): h = self.hash(key) for k, v in self.table[h]: if k == key: return v return None ```

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

星*湖

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值