高效查找算法详解:从顺序到哈希

一、今天学习目标

  1. 顺序查找
  2. 二分查找(折半查找)必考
  3. 插值查找、斐波那契查找(了解思路)
  4. 哈希查找
  5. 各种查找对比 + 适用场景

二、1. 顺序查找(线性查找)

逐个遍历,直到找到。

  • 优点:不要求数组有序
  • 缺点:慢 O (n)
cpp 复制代码
int seqSearch(int arr[], int n, int key) {
    for (int i = 0; i < n; i++) {
        if (arr[i] == key)
            return i;
    }
    return -1;
}

三、2. 二分查找(折半查找)⭐⭐⭐⭐⭐

必须有序!必须有序!必须有序!

思想:每次取中间,缩小一半查找范围。时间复杂度 O(log n)

cpp 复制代码
// 二分查找(迭代版,最常用)
int binarySearch(int arr[], int n, int key) {
    int l = 0, r = n - 1;
    while (l <= r) {
        int mid = l + (r - l) / 2;  // 防溢出
        if (arr[mid] == key)
            return mid;
        else if (arr[mid] < key)
            l = mid + 1;
        else
            r = mid - 1;
    }
    return -1;
}

四、3. 插值查找

按比例猜位置:mid = l + (key - arr[l]) * (r - l) / (arr[r] - arr[l])

  • 数据分布均匀时比二分更快
  • 极端数据反而慢

五、4. 斐波那契查找

利用斐波那契数列分割区间思路接近二分,只是分割点不同。了解即可,面试基本不手写。


六、5. 哈希查找

通过哈希函数直接定位平均 O (1),无序也能查就是我们 day17 实现的哈希表。


七、完整测试代码

cpp 复制代码
#include <stdio.h>

int seqSearch(int arr[], int n, int key) {
    for (int i = 0; i < n; i++)
        if (arr[i] == key) return i;
    return -1;
}

int binarySearch(int arr[], int n, int key) {
    int l = 0, r = n - 1;
    while (l <= r) {
        int mid = l + (r - l) / 2;
        if (arr[mid] == key) return mid;
        else if (arr[mid] < key) l = mid + 1;
        else r = mid - 1;
    }
    return -1;
}

void printRes(int pos, int key) {
    if (pos != -1)
        printf("找到 %d,下标:%d\n", key, pos);
    else
        printf("未找到 %d\n", key);
}

int main() {
    int arr[] = {1, 3, 5, 7, 9, 11, 13};
    int n = sizeof(arr)/sizeof(arr[0]);
    int key = 7;

    printRes(seqSearch(arr, n, key), key);
    printRes(binarySearch(arr, n, key), key);
    return 0;
}

运行结果:

复制代码
找到 7,下标:3
找到 7,下标:3

八、查找算法对比表(面试背这个)

表格

查找 数据要求 时间复杂度 优点
顺序查找 无序有序均可 O(n) 简单,不挑数据
二分查找 必须有序 O(log n) 极快,最常用
插值查找 有序且分布均匀 O(log log n) 均匀数据更快
斐波那契查找 有序 O(log n) 适合嵌入式
哈希查找 无序 O (1) 平均 最快,工程首选

九、今日小练习

有序数组:[2,4,6,8,10,12,14,16]

  1. 用二分查找找 10
  2. 用二分查找找 5
  3. 输出下标或 -1
相关推荐
不是光头 强10 小时前
feign-list-param-crash-cpp
java·数据结构·list
x_xbx10 小时前
LeetCode:20. 有效的括号
算法·leetcode·职场和发展
计算机安禾10 小时前
【算法设计与分析】第40篇:空间数据结构:KD树与四叉树的查询分析
数据结构·算法
努力努力再努力wz10 小时前
【C++高阶数据结构系列】:跳表 SkipList 详解:多层索引、随机晋升与C++ 完整实现(附跳表实现的源码)
开发语言·数据结构·数据库·c++·redis·缓存·skiplist
m沐沐10 小时前
【机器学习】信用卡欺诈检测实战:逻辑回归 + 过采样
人工智能·算法·机器学习·pycharm·逻辑回归
代码中介商10 小时前
图论入门:从基础到遍历算法
数据结构·算法·图论
csdn_aspnet10 小时前
Python 霍尔分区算法(Hoare‘s Partition Algorithm)
开发语言·python·算法
8Qi810 小时前
LeetCode 295:数据流的中位数(Median Finder)—— Java 题解 ✅
java·算法·leetcode·优先队列··中位数
Trouvaille ~10 小时前
【Redis篇】Redis 事务:原子性与脚本执行机制
数据库·redis·后端·算法·junit·lua·原子性
飞天狗11110 小时前
2024第十五届蓝桥杯c/c++B组国赛题解
c语言·数据结构·c++·算法·蓝桥杯