”工欲善其事,必先利其器。“—孔子《论语.录灵公》
首页 > 编程 > “警惕时间复杂度陷阱”

“警惕时间复杂度陷阱”

发布于2024-08-16
浏览:732

“Be wary of time complexity pitfalls\

警惕时间复杂度陷阱

写在这里

bilibili视频也显示了这个:[Bilibili视频][https://www.bilibili.com/video/BV16u4m1c7cU/?spm_id_from=333.999.0.0]我认为这是一个很好的视频,但它的语言是中文。

时间复杂度

  • 时间复杂度是什么意思?
  • 时间复杂度是算法运行所需时间的度量,作为其输入大小的函数。它是描述算法效率的一种方式,用于比较不同的算法并确定哪种算法最有效。

  • 如何计算时间复杂度?

  • 为了计算时间复杂度,我们需要考虑算法执行的操作数作为输入大小的函数。测量操作次数的最常见方法是计算执行特定操作的次数。

  • 计算时间复杂度时有哪些常见陷阱?

    1. 不考虑输入大小:如果我们只考虑算法执行的操作数量,我们可能会低估时间复杂度。例如,如果我们计算循环运行的次数,但不考虑输入的大小,那么时间复杂度可能会太高。
    1. 不考虑算法的效率:有些算法即使输入量很小也可能执行很多操作,这可能导致时间复杂度很高。例如,冒泡排序和选择排序等排序算法具有二次时间复杂度,即使它们可能只需要交换小数组中的两个元素。
    1. 不考虑算法的最坏情况:某些算法具有最好情况,其中执行的操作比最坏情况要少。例如,像二分搜索这样的搜索算法有一个最好的情况,即它们在对数时间内找到目标元素,但它们有一个最坏的情况,即它们需要检查数组中的所有元素。

让我们看一些时间复杂度的例子

这里有一个问题:
查找列表中最多 10 个整数。

import random
ls = [random.randint(1, 100) for _ in range(n)]

这里是测试函数:

import time
def benchmark(func, *args) -> float:
    start = time.perf_counter()
    func(*args)
    end = time.perf_counter()
    return end - start

解决方案1:使用堆

这是使用heapq模块的解决方案:

def find_max_n(ls, n):
    import heapq
    return heapq.nlargest(n, ls)

或者我们用python的方式写:

def find_largest_n(nums, n):
    if n  max_heap[0]:
            max_heap[0] = num
            _sift_down(max_heap, 0)

    return max_heap

def _sift_down(heap, index):
    left = 2 * index   1
    right = 2 * index   2
    largest = index

    if left  heap[largest]:
        largest = left

    if right  heap[largest]:
        largest = right

    if largest != index:
        heap[index], heap[largest] = heap[largest], heap[index]
        _sift_down(heap, largest)

解决方案2:使用排序

这里是使用排序功能的解决方案:

def find_max_n(ls, n):
    return sorted(ls, reverse=True)[:n]

几乎所有人都知道,堆的时间复杂度是 O(n log k),排序函数的时间复杂度是 O(n log n)。

看来第一个解决方案比第二个更好。但在Python中却并非如此。

看最终代码:

import time
def benchmark(func, *args) -> float:
    start = time.perf_counter()
    func(*args)
    end = time.perf_counter()
    return end - start

def find_max_n_heapq(ls, n):
    import heapq
    return heapq.nlargest(n, ls)

def find_max_n_python_heap(nums, n):
    if n  max_heap[0]:
            max_heap[0] = num
            _sift_down(max_heap, 0)

    return max_heap

def _sift_down(heap, index):
    left = 2 * index   1
    right = 2 * index   2
    largest = index

    if left  heap[largest]:
        largest = left

    if right  heap[largest]:
        largest = right

    if largest != index:
        heap[index], heap[largest] = heap[largest], heap[index]
        _sift_down(heap, largest)


def find_max_n_sorted(ls, n):
    return sorted(ls, reverse=True)[:n]

# test
import random
for n in range(10, 10000, 100):
    ls = [random.randint(1, 100) for _ in range(n)]
    print(f"n = {n}")
    print(f"Use    Heapq: {benchmark(find_max_n_heapq, ls, n)}")
    print(f"Python Heapq: {benchmark(find_max_n_python_heap, ls, n)}")
    print(f"Sorted      : {benchmark(find_max_n_sorted, ls, n)}")

我在Python3.11 VScode中运行,结果如下:

n 不大

使用堆:0.002430099993944168
Python堆:0.06343129999004304
排序:9.280000813305378e-05
n = 910
使用堆:9.220000356435776e-05
Python堆:0.07715500006452203
排序:9.360001422464848e-05
n = 920
使用堆:9.440002031624317e-05
Python堆:0.06573969998862594
排序:0.00012450001668184996
n = 930
使用堆:9.689992293715477e-05
Python堆:0.06760239996947348
已排序:9.66999214142561e-05
n = 940
使用堆:9.600003249943256e-05
Python堆:0.07372559991199523
排序:9.680003859102726e-05
n = 950
使用堆:9.770004544407129e-05
Python堆:0.07306570000946522
排序:0.00011979998089373112
n = 960
使用堆:9.980006143450737e-05
Python堆:0.0771204000338912
排序:0.00022829999215900898
n = 970
使用堆:0.0001601999392732978
Python堆:0.07493270002305508
排序:0.00010840001050382853
n = 980
使用堆:9.949994273483753e-05
Python 堆:0.07698320003692061
排序:0.00010300008580088615
n = 990
使用堆:9.979994501918554e-05
Python堆:0.0848745999392122
排序:0.00012620002962648869

如果n很大?

n = 10000
使用堆:0.003642000025138259
Python 堆:9.698883199947886
排序:0.00107999995816499
n = 11000
使用堆:0.0014836000045761466
Python 堆:10.537632800056599
排序:0.0012236000038683414
n = 12000
使用堆:0.001384599949233234
Python 堆:12.328411899972707
排序:0.0013226999435573816
n = 13000
使用堆:0.0020017001079395413
Python 堆:15.637207800056785
排序:0.0015075999544933438
n = 14000
使用堆:0.0017026999266818166
Python 堆:17.298848500009626
排序:0.0016967999981716275
n = 15000
使用堆:0.0017773000290617347
Python 堆:20.780625900020823
排序:0.0017105999868363142

我发现了什么以及如何改进它

当n很大时,Sorted会花费一点时间(有时甚至比使用heapq更好),但Python Heapq会花费很多时间。

  • 为什么Sorted花费一点时间,而Python Heapq花费很多时间?
  • 因为sorted()是Python中的内置函数,所以你可以找到关于它的Python官方文档。

内置函数比 heapq 更快,因为它是用 C 编写的,C 是一种编译语言。

  • 如何改进?
  • 您可以使用内置函数sorted()代替heapq.sort()来提高代码的性能。 Sorted() 函数是 Python 中的内置函数,它是用 C 实现的,因此比 heapq.sort()
  • 快得多

脑震荡

当我们处理大数据时,我们应该使用内置函数而不是 heapq.sort() 来提高代码的性能。在处理大数据时,我们必须警惕时间复杂度陷阱。有时时间复杂度的陷阱是不可避免的,但我们应该尽量避免它们。

关于我

大家好,我是梦沁园。我是一名学生。我喜欢学习新事物。
你可以看我的github:[MengQinYuan的Github][https://github.com/mengqinyuan]

版本声明 本文转载于:https://dev.to/mengqinyuan/be-wary-of-time-complexity-pitfalls-13jf?1如有侵犯,请联系[email protected]删除
最新教程 更多>
  • 如何在 Golang Web 服务器中流式传输 MP4 视频?
    如何在 Golang Web 服务器中流式传输 MP4 视频?
    GoLang Web 服务器流视频问:Golang Web 服务器配置为服务 HTML、CSS、JavaScript 和图像失败尝试流式传输 MP4 视频。该问题是由于处理大型视频文件而引起的。 Chrome 要求服务器支持超过一定大小的视频的 Range 请求,但提供的代码没有解决这个问题。通过实...
    编程 发布于2024-11-14
  • Bootstrap 4 Beta 中的列偏移发生了什么?
    Bootstrap 4 Beta 中的列偏移发生了什么?
    Bootstrap 4 Beta:列偏移的删除和恢复Bootstrap 4 在其 Beta 1 版本中引入了重大更改柱子偏移了。然而,随着 Beta 2 的后续发布,这些变化已经逆转。从 offset-md-* 到 ml-auto在 Bootstrap 4 Beta 1 中, offset-md-*...
    编程 发布于2024-11-14
  • 如何修复 macOS 上 Django 中的“配置不正确:加载 MySQLdb 模块时出错”?
    如何修复 macOS 上 Django 中的“配置不正确:加载 MySQLdb 模块时出错”?
    MySQL配置不正确:相对路径的问题在Django中运行python manage.py runserver时,可能会遇到以下错误:ImproperlyConfigured: Error loading MySQLdb module: dlopen(/Library/Python/2.7/site-...
    编程 发布于2024-11-14
  • 如何使用 MySQL 查找今天生日的用户?
    如何使用 MySQL 查找今天生日的用户?
    如何使用 MySQL 识别今天生日的用户使用 MySQL 确定今天是否是用户的生日涉及查找生日匹配的所有行今天的日期。这可以通过一个简单的 MySQL 查询来实现,该查询将存储为 UNIX 时间戳的生日与今天的日期进行比较。以下 SQL 查询将获取今天有生日的所有用户: FROM USERS ...
    编程 发布于2024-11-14
  • CSS 中的“display: table-column”实际上做了什么?
    CSS 中的“display: table-column”实际上做了什么?
    CSS“display: table-column”应该如何工作?在 HTML 中,表格由行组成,每行含有细胞。 CSS 扩展了这个概念,允许设计者定义特定的行和列布局。虽然“display: table-row”和“display: table-cell”很简单,但“display: table-...
    编程 发布于2024-11-14
  • Babel 6 如何以不同的方式处理默认导出?
    Babel 6 如何以不同的方式处理默认导出?
    重大变更:Babel 6 导出默认行为随着 Babel 6 的发布,默认导出的处理方式发生了重大变化。虽然 Babel 之前添加了 module.exports = Exports["default"] 行,但此功能已被删除。此修改需要更改模块导入语法。以前,使用旧语法的代码:v...
    编程 发布于2024-11-14
  • 掌握 Next.js 中的 SSR:如何提升 SEO 和用户体验
    掌握 Next.js 中的 SSR:如何提升 SEO 和用户体验
    SSR(服务器端渲染)是 Next.js 中生成页面的另一种方法。在本文中,我想解释什么是 SSR、它是如何工作的,以及如何在 Next.js 项目的 Page Router 和 App Router 中实现它。 什么是SSR? SSR是一种在用户发出请求后生成静态页面(或预渲染页面...
    编程 发布于2024-11-14
  • 为什么 PHP 5.2 不允许抽象静态类方法?
    为什么 PHP 5.2 不允许抽象静态类方法?
    PHP 5.2 严格模式:为什么不允许抽象静态类方法?在 PHP 5.2 中,启用严格警告可能会触发熟悉的警告:“静态函数不应该是抽象的”。此警告源于 PHP 5.2 中引入的一项更改,该更改不允许抽象静态类方法。原因:历史监督PHP 5.2 最初缺乏后期静态绑定,使抽象静态函数变得无用。由于抽象静...
    编程 发布于2024-11-14
  • 如何为 10 个连续点的每段绘制不同颜色的线?
    如何为 10 个连续点的每段绘制不同颜色的线?
    用不同的颜色绘制一条线问题陈述给定两个列表,latt和lont,目标是绘制一条线,其中每个列表10 个连续点的线段以不同的方式表示color.解决方案线段数量有限如果线段数量较少,例如10个或更少,一个简单的方法就是使用循环以唯一的颜色绘制每个段。import numpy as np import ...
    编程 发布于2024-11-14
  • 如何在 MySQL 中根据计数过滤数据而不使用嵌套 SELECT?
    如何在 MySQL 中根据计数过滤数据而不使用嵌套 SELECT?
    MySQL - 在 WHERE 子句中使用 COUNT(*)用户在尝试使用 WHERE 子句中的 COUNT(*) 函数过滤 MySQL 中的数据时遇到了挑战WHERE 子句。他们寻求一种有效的方法来完成此任务,而不使用嵌套 SELECT 语句,因为它会消耗大量资源。用户提供了以下伪代码来说明他们期...
    编程 发布于2024-11-14
  • 如何在 Python 中按名称访问 SQL 结果列值?
    如何在 Python 中按名称访问 SQL 结果列值?
    在 Python 中按列名称访问 SQL 结果列值处理数据库中的大量列时,依赖于列索引数据检索可能会变得很麻烦。本文通过提供一种在 Python 中使用列名称检索 SQL 结果列值的方法来解决对更直观方法的需求。解决方案:利用 DictCursor Python 的 MySQLdb 模块提供了 Di...
    编程 发布于2024-11-14
  • 除了“if”语句之外:还有哪些地方可以在不进行强制转换的情况下使用具有显式“bool”转换的类型?
    除了“if”语句之外:还有哪些地方可以在不进行强制转换的情况下使用具有显式“bool”转换的类型?
    无需强制转换即可上下文转换为 bool您的类定义了对 bool 的显式转换,使您能够在条件语句中直接使用其实例“t”。然而,这种显式转换提出了一个问题:“t”在哪里可以在不进行强制转换的情况下用作 bool?上下文转换场景C 标准指定了四种值可以根据上下文转换为的主要场景bool:语句:if、whi...
    编程 发布于2024-11-14
  • 何时使用 Django ORM 的 select_lated 与 prefetch_lated?
    何时使用 Django ORM 的 select_lated 与 prefetch_lated?
    Django ORM 的 select_lated 和 prefetch_lated 之间的区别在 Django ORM 中,select_lated 和 prefetch_lated 方法在管理数据库查询中的关系方面具有不同的用途。select_latedDjango的select_lated方法...
    编程 发布于2024-11-14
  • 大批
    大批
    方法是可以在对象上调用的 fns 数组是对象,因此它们在 JS 中也有方法。 slice(begin):将数组的一部分提取到新数组中,而不改变原始数组。 let arr = ['a','b','c','d','e']; // Usecase: Extract till index p...
    编程 发布于2024-11-14
  • 在 Go 中使用 WebSocket 进行实时通信
    在 Go 中使用 WebSocket 进行实时通信
    构建需要实时更新的应用程序(例如聊天应用程序、实时通知或协作工具)需要一种比传统 HTTP 更快、更具交互性的通信方法。这就是 WebSockets 发挥作用的地方!今天,我们将探讨如何在 Go 中使用 WebSocket,以便您可以向应用程序添加实时功能。 在这篇文章中,我们将介绍: WebSoc...
    编程 发布于2024-11-13

免责声明: 提供的所有资源部分来自互联网,如果有侵犯您的版权或其他权益,请说明详细缘由并提供版权或权益证明然后发到邮箱:[email protected] 我们会第一时间内为您处理。

Copyright© 2022 湘ICP备2022001581号-3