搜索
首页后端开发Python教程需要修复的常见 Django ORM 错误

Common Django ORM Mistakes to fix

Django ORM 是 django 最强大的功能之一。它抽象化了与数据库交互的大部分复杂性,让开发人员可以使用 Pythonic 语法而不是原始 SQL 来操作数据。所有这些 ORM 函数都会生成 SQL 查询,如果处理不当,这些查询可能会成为瓶颈。
本博客重点介绍了使用 Django ORM 时的常见错误,并提供了保持查询高效、可维护和高性能的技巧。

1. N 1 查询问题

当您的代码触发一个查询来获取一组记录,然后再次运行 N 个附加查询来获取相关数据时,就会出现 N 1 查询问题。

blogs = Blog.objects.all()    # 1 Query
for blog in blogs:
    print(blog.author.name)   # N additional queries

在上面的示例中,在循环内访问 blog.author.name 会导致 Django 单独获取每个博客的作者记录,从而导致 N 个额外的查询。

如何解决
对单个相关对象(例如,ForeignKey 或 OneToOneField)使用 select_lated,因为它执行 SQL JOIN 以在一个查询中检索主对象及其相关对象。对于多对多、多对一或反向关系,请使用 prefetch_lated,它在单独的查询中获取相关数据,但在 Python 中有效地将它们组合起来,避免 N 1 问题。

# With select_related
blogs = Blog.objects.select_related('author').all()

# With prefetch_related
authors = Author.objects.prefetch_related('blogs').all()

2. 过度使用.all()和.filter()

开发人员经常链接多个过滤器或使用 .all() ,然后对同一查询集重复查询:

blogs = Blog.objects.all()
active_blogs = blogs.filter(is_archived=False)
popular_blogs = blogs.filter(views__gte=1000)

尽管 Django 尝试通过仅在需要时延迟评估查询集来优化查询集,但对同一查询集数据重复调用过滤器仍然会导致对数据库不必要的命中。

如何解决
在一条语句中组合过滤器允许 django 生成单个 SQL 查询。

popular_active_blogs = Blog.objects.filter(is_archived=False, views__gte=1000)

3. 不利用values()或values_list()

有时我们只需要特定字段而不是模型的所有字段数据。在此期间使用 .values().values_list() 可以更高效。

titles = Blog.objects.values('title')
or
titles = Blog.objects.values_list('title', flat=True)
# values() returns a list of dictionaries.
# values_list() can return tuples or flat values if flat=True is provided.

通过仅获取所需的列,可以减少从数据库传输的数据量,从而提高性能。

4. 低效的聚合和注释

重复调用 .aggregate().annotate() 可能会导致多次查询。具有多个注释的复杂查询可能会导致 SQL 查询效率低下,从而可能导致繁重的数据库操作。

# Example of multiple aggregate
total_count = Blog.objects.aggregate(Count('id'))
author_count = Blog.objects.aggregate(Count('author'))
average_views = Blog.objects.aggregate(Avg('views'))

推荐

blogs = Blog.objects.all()    # 1 Query
for blog in blogs:
    print(blog.author.name)   # N additional queries

5. 不使用数据库索引

索引使数据库能够快速定位和检索数据,避免缓慢的全表扫描,从而提高查询性能。索引优化了过滤、排序和连接等操作,使得对频繁访问的字段的查询速度更快。频繁查询字段上缺少数据库索引会大大降低性能。
如何在 Django 中添加索引

# With select_related
blogs = Blog.objects.select_related('author').all()

# With prefetch_related
authors = Author.objects.prefetch_related('blogs').all()

索引可以加快读取速度,但会减慢写入速度。因此,只对那些经常需要查询的字段建立索引。

6. 不使用缓存

当我们必须查询计算成本昂贵或很少更改的数据时,请使用缓存。即使缓存 5 分钟,也可以节省重复查询、复杂计算和不经常更改的查询。

blogs = Blog.objects.all()
active_blogs = blogs.filter(is_archived=False)
popular_blogs = blogs.filter(views__gte=1000)

7. 原始 SQL

有时,Django ORM 无法有效地表达复杂的查询或批量操作。虽然 Django 提供 .extra() 或 .raw(),但原始 SQL 使用应该是最后的手段,因为它:

  • 失去了 ORM 的许多好处
  • 可能导致不可读或容易出错的代码

确保输入得到正确清理并保持原始 SQL 查询可维护。

应用这些技巧,您将提高 Django 应用程序的性能,同时保持代码整洁和可维护。并建议在开发环境中使用 Django 调试工具栏 来监控和分析执行的查询数量、执行时间和 SQL 语句。

以上是需要修复的常见 Django ORM 错误的详细内容。更多信息请关注PHP中文网其他相关文章!

声明
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn
您如何将元素附加到Python数组?您如何将元素附加到Python数组?Apr 30, 2025 am 12:19 AM

Inpython,YouAppendElementStoAlistusingTheAppend()方法。1)useappend()forsingleelements:my_list.append(4).2)useextend()orextend()或= formultiplelements:my_list.extend.extend(emote_list)ormy_list = [4,5,6] .3)useInsert()forspefificpositions:my_list.insert(1,5).beaware

您如何调试与Shebang有关的问题?您如何调试与Shebang有关的问题?Apr 30, 2025 am 12:17 AM

调试shebang问题的方法包括:1.检查shebang行确保是脚本首行且无前置空格;2.验证解释器路径是否正确;3.直接调用解释器运行脚本以隔离shebang问题;4.使用strace或truss跟踪系统调用;5.检查环境变量对shebang的影响。

如何从python数组中删除元素?如何从python数组中删除元素?Apr 30, 2025 am 12:16 AM

pythonlistscanbemanipulationusesseveralmethodstoremovelements:1)theremove()MethodRemovestHefirStocCurrenceOfAstePecifiedValue.2)thepop()thepop()methodremovesandremovesandurturnturnsananelementatagivenIndex.3)

可以在Python列表中存储哪些数据类型?可以在Python列表中存储哪些数据类型?Apr 30, 2025 am 12:07 AM

pythonlistscanstoreanydatate型,包括素,弦,浮子,布尔人,其他列表和迪克尼亚式

在Python列表上可以执行哪些常见操作?在Python列表上可以执行哪些常见操作?Apr 30, 2025 am 12:01 AM

pythristssupportnumereperations:1)addingElementSwithAppend(),Extend(),andInsert()。2)emovingItemSusingRemove(),pop(),andclear(),and clear()。3)访问andmodifyingandmodifyingwithIndexingAndexingAndSlicing.4)

如何使用numpy创建多维数组?如何使用numpy创建多维数组?Apr 29, 2025 am 12:27 AM

使用NumPy创建多维数组可以通过以下步骤实现:1)使用numpy.array()函数创建数组,例如np.array([[1,2,3],[4,5,6]])创建2D数组;2)使用np.zeros(),np.ones(),np.random.random()等函数创建特定值填充的数组;3)理解数组的shape和size属性,确保子数组长度一致,避免错误;4)使用np.reshape()函数改变数组形状;5)注意内存使用,确保代码清晰高效。

说明Numpy阵列中'广播”的概念。说明Numpy阵列中'广播”的概念。Apr 29, 2025 am 12:23 AM

播放innumpyisamethodtoperformoperationsonArraySofDifferentsHapesbyAutapityallate AligningThem.itSimplifififiesCode,增强可读性,和Boostsperformance.Shere'shore'showitworks:1)较小的ArraySaraySaraysAraySaraySaraySaraySarePaddedDedWiteWithOnestOmatchDimentions.2)

说明如何在列表,Array.Array和用于数据存储的Numpy数组之间进行选择。说明如何在列表,Array.Array和用于数据存储的Numpy数组之间进行选择。Apr 29, 2025 am 12:20 AM

forpythondataTastorage,choselistsforflexibilityWithMixedDatatypes,array.ArrayFormeMory-effficityHomogeneousnumericalData,andnumpyArraysForAdvancedNumericalComputing.listsareversareversareversareversArversatilebutlessEbutlesseftlesseftlesseftlessforefforefforefforefforefforefforefforefforefforlargenumerdataSets; arrayoffray.array.array.array.array.array.ersersamiddreddregro

See all articles

热AI工具

Undresser.AI Undress

Undresser.AI Undress

人工智能驱动的应用程序,用于创建逼真的裸体照片

AI Clothes Remover

AI Clothes Remover

用于从照片中去除衣服的在线人工智能工具。

Undress AI Tool

Undress AI Tool

免费脱衣服图片

Clothoff.io

Clothoff.io

AI脱衣机

Video Face Swap

Video Face Swap

使用我们完全免费的人工智能换脸工具轻松在任何视频中换脸!

热工具

Dreamweaver Mac版

Dreamweaver Mac版

视觉化网页开发工具

SublimeText3 英文版

SublimeText3 英文版

推荐:为Win版本,支持代码提示!

SublimeText3 Mac版

SublimeText3 Mac版

神级代码编辑软件(SublimeText3)

VSCode Windows 64位 下载

VSCode Windows 64位 下载

微软推出的免费、功能强大的一款IDE编辑器

禅工作室 13.0.1

禅工作室 13.0.1

功能强大的PHP集成开发环境