搜尋
首頁後端開發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 Switch語句?什麼是Python Switch語句?Apr 30, 2025 pm 02:08 PM

本文討論了版本3.10中介紹的Python的新“匹配”語句,該語句與其他語言相同。它增強了代碼的可讀性,並為傳統的if-elif-el提供了性能優勢

Python中有什麼例外組?Python中有什麼例外組?Apr 30, 2025 pm 02:07 PM

Python 3.11中的異常組允許同時處理多個異常,從而改善了並發方案和復雜操作中的錯誤管理。

Python中的功能註釋是什麼?Python中的功能註釋是什麼?Apr 30, 2025 pm 02:06 PM

Python中的功能註釋將元數據添加到函數中,以進行類型檢查,文檔和IDE支持。它們增強了代碼的可讀性,維護,並且在API開發,數據科學和圖書館創建中至關重要。

Python的單位測試是什麼?Python的單位測試是什麼?Apr 30, 2025 pm 02:05 PM

本文討論了Python中的單位測試,其好處以及如何有效編寫它們。它突出顯示了諸如UNITSEST和PYTEST之類的工具進行測試。

Python中的訪問說明符是什麼?Python中的訪問說明符是什麼?Apr 30, 2025 pm 02:03 PM

文章討論了Python中的訪問說明符,這些說明符使用命名慣例表明班級成員的可見性,而不是嚴格的執法。

Python中的__Init __()是什麼?自我如何在其中發揮作用?Python中的__Init __()是什麼?自我如何在其中發揮作用?Apr 30, 2025 pm 02:02 PM

文章討論了Python的\ _ \ _ Init \ _ \ _()方法和Self在初始化對象屬性中的作用。還涵蓋了其他類方法和繼承對\ _ \ _ Init \ _ \ _()的影響。

python中的@classmethod,@staticmethod和實例方法有什麼區別?python中的@classmethod,@staticmethod和實例方法有什麼區別?Apr 30, 2025 pm 02:01 PM

本文討論了python中@classmethod,@staticmethod和實例方法之間的差異,詳細介紹了它們的屬性,用例和好處。它說明瞭如何根據所需功能選擇正確的方法類型和DA

您如何將元素附加到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

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

使用我們完全免費的人工智慧換臉工具,輕鬆在任何影片中換臉!

熱工具

ZendStudio 13.5.1 Mac

ZendStudio 13.5.1 Mac

強大的PHP整合開發環境

mPDF

mPDF

mPDF是一個PHP庫,可以從UTF-8編碼的HTML產生PDF檔案。原作者Ian Back編寫mPDF以從他的網站上「即時」輸出PDF文件,並處理不同的語言。與原始腳本如HTML2FPDF相比,它的速度較慢,並且在使用Unicode字體時產生的檔案較大,但支援CSS樣式等,並進行了大量增強。支援幾乎所有語言,包括RTL(阿拉伯語和希伯來語)和CJK(中日韓)。支援嵌套的區塊級元素(如P、DIV),

SecLists

SecLists

SecLists是最終安全測試人員的伙伴。它是一個包含各種類型清單的集合,這些清單在安全評估過程中經常使用,而且都在一個地方。 SecLists透過方便地提供安全測試人員可能需要的所有列表,幫助提高安全測試的效率和生產力。清單類型包括使用者名稱、密碼、URL、模糊測試有效載荷、敏感資料模式、Web shell等等。測試人員只需將此儲存庫拉到新的測試機上,他就可以存取所需的每種類型的清單。

SAP NetWeaver Server Adapter for Eclipse

SAP NetWeaver Server Adapter for Eclipse

將Eclipse與SAP NetWeaver應用伺服器整合。

MinGW - Minimalist GNU for Windows

MinGW - Minimalist GNU for Windows

這個專案正在遷移到osdn.net/projects/mingw的過程中,你可以繼續在那裡關注我們。 MinGW:GNU編譯器集合(GCC)的本機Windows移植版本,可自由分發的導入函式庫和用於建置本機Windows應用程式的頭檔;包括對MSVC執行時間的擴展,以支援C99功能。 MinGW的所有軟體都可以在64位元Windows平台上運作。