浅析折半查找的时间复杂度

一、折半查找的定义

折半查找(二分查找)是一种常用的查找算法,它要求待查找的数据结构必须是有序的。

其实现过程如下:

int binary_search(int arr[], int left, int right, int target)
{
    while(left <= right)
    {
        int mid = (left + right) / 2;
        if(arr[mid] == target)
            return mid;
        if(arr[mid] < target)
            left = mid + 1;
        else
            right = mid - 1;
    }
    return -1;
}

二、折半查找的时间复杂度与时间复杂度分析

常见的算法时间复杂度分别有O(1), O(logn), O(n), O(nlogn), O(n²), O(n³)等。

而折半查找的时间复杂度为O(logn),也就是说它属于时间复杂度比较小的算法,适用于大数据量的快速查找。

下面从多个方面来详细阐述折半查找的时间复杂度。

三、折半查找的时间复杂度与数据规模

折半查找的时间复杂度与数据规模有关,具体表现为:

当数据规模n增大时,时间复杂度O(logn)的速度要比O(n)快,因为O(logn)的增长速度是比O(n)慢的。

如下面的示例所示,随着数据规模的增大,折半查找算法与普通查找算法的时间复杂度对比:

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

int search(int arr[], int len, int target)
{
    for(int i = 0; i < len; i++)
    {
        if(arr[i] == target)
            return i;
    }
    return -1;
}

int binary_search(int arr[], int left, int right, int target)
{
    while(left <= right)
    {
        int mid = (left + right) / 2;
        if(arr[mid] == target)
            return mid;
        if(arr[mid] < target)
            left = mid + 1;
        else
            right = mid - 1;
    }
    return -1;
}

int main()
{
    int len = 100000000;
    int* arr = (int*)malloc(len * sizeof(int));
    for(int i = 0; i < len; i++)
        arr[i] = i;
    int target = len - 1;
    clock_t start, end;
    int ret;

    start = clock();
    ret = binary_search(arr, 0, len - 1, target);
    end = clock();
    printf("binary_search: %d, time: %fs\n", ret, (double)(end - start) / CLOCKS_PER_SEC);

    start = clock();
    ret = search(arr, len, target);
    end = clock();
    printf("search: %d, time: %fs\n", ret, (double)(end - start) / CLOCKS_PER_SEC);

    free(arr);
    return 0;
}

当数据规模为100000000时,折半查找算法的耗时只有0.001秒,而普通查找算法的耗时却为4.737秒。

四、折半查找的时间复杂度与查找时间

在平均时间复杂度为O(logn)的前提下,折半查找算法的查找时间比较快,因为它每次可以将查找范围缩小为原来的一半。

而在最坏情况下,折半查找的时间复杂度O(logn)也会失效,比如当查找的关键字不在序列中时,折半查找会一直循环到我查找范围为空,这时它的时间复杂度将达到O(n)。

下面的示例程序中,为了让折半查找失效,我将查找目标设置为0xFFFFFFF(因为序列中最大的数为0x7FFFFFFF,所以它不可能在序列中):

int binary_search(int arr[], int left, int right, int target)
{
    while(left <= right)
    {
        int mid = (left + right) / 2;
        if(arr[mid] == target)
            return mid;
        if(arr[mid] < target)
            left = mid + 1;
        else
            right = mid - 1;
    }
    return -1;
}

int main()
{
    int arr[] = {1,2,3,4,5,6,7,8,9,10};
    int ret;

    ret = binary_search(arr, 0, 9, 0xFFFFFFF);
    printf("%d\n", ret);

    return 0;
}

程序会输出-1,表示查找目标不在序列中。由于这个查找目标不在序列中,折半查找算法的每次循环都会扫描整个序列,这时时间复杂度会退化为O(n)。

总结

综上所述,折半查找算法是一种时间复杂度为O(logn)的快速查找算法。在数据规模较大的情况下,折半查找算法明显比普通查找算法快许多。但在最坏情况下,它的时间复杂度会失效,这时需要考虑到其他查找算法。

原创文章,作者:小蓝,如若转载,请注明出处:https://www.506064.com/n/234007.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
小蓝小蓝
上一篇 2024-12-11 17:12
下一篇 2024-12-11 17:12

相关推荐

  • 解决docker-compose 容器时间和服务器时间不同步问题

    docker-compose是一种工具,能够让您使用YAML文件来定义和运行多个容器。然而,有时候容器的时间与服务器时间不同步,导致一些不必要的错误和麻烦。以下是解决方法的详细介绍…

    编程 2025-04-29
  • 想把你和时间藏起来

    如果你觉得时间过得太快,每天都过得太匆忙,那么你是否曾经想过想把时间藏起来,慢慢享受每一个瞬间?在这篇文章中,我们将会从多个方面,详细地阐述如何想把你和时间藏起来。 一、一些时间管…

    编程 2025-04-28
  • 计算斐波那契数列的时间复杂度解析

    斐波那契数列是一个数列,其中每个数都是前两个数的和,第一个数和第二个数都是1。斐波那契数列的前几项为:1,1,2,3,5,8,13,21,34,…。计算斐波那契数列常用…

    编程 2025-04-28
  • 时间戳秒级可以用int吗

    时间戳是指从某个固定的时间点开始计算的已经过去的时间。在计算机领域,时间戳通常使用秒级或毫秒级来表示。在实际使用中,我们经常会遇到需要将时间戳转换为整数类型的情况。那么,时间戳秒级…

    编程 2025-04-28
  • 如何在ACM竞赛中优化开发时间

    ACM竞赛旨在提高程序员的算法能力和解决问题的实力,然而在比赛中优化开发时间同样至关重要。 一、规划赛前准备 1、提前熟悉比赛规则和题目类型,了解常见算法、数据结构和快速编写代码的…

    编程 2025-04-28
  • 使用JavaScript日期函数掌握时间

    在本文中,我们将深入探讨JavaScript日期函数,并且从多个视角介绍其应用方法和重要性。 一、日期的基本表示与获取 在JavaScript中,使用Date对象来表示日期和时间,…

    编程 2025-04-28
  • Java Date时间大小比较

    本文将从多个角度详细阐述Java中Date时间大小的比较,包含了时间字符串转换、日期相减、使用Calendar比较、使用compareTo方法比较等多个方面。相信这篇文章能够对你解…

    编程 2025-04-27
  • 从时间复杂度角度看循环赛日程表

    循环赛日程表是指在一个比赛中,每个参赛者都需要与其他所有参赛者逐一比赛一次,而且每个参赛者可以在同一场比赛中和其他参赛者比赛多次,比如足球、篮球等。循环赛日程表的设计需要考虑时间复…

    编程 2025-04-27
  • 二分查找时间复杂度为什么是logN – 知乎

    二分查找是一种常用的查找算法。它通过将目标值与数组的中间元素进行比较,从而将查找范围缩小一半,直到找到目标值。这种方法的时间复杂度为O(logN)。下面我们将从多个方面探讨为什么二…

    编程 2025-04-27
  • One change 时间:简化项目开发的最佳实践

    本文将介绍 One change 时间 (OCT) 的定义和实现方法,并探讨它如何简化项目开发。OCT 是一种项目开发和管理的策略,通过将更改限制在固定的时间间隔(通常为一周)内,…

    编程 2025-04-27

发表回复

登录后才能评论