主頁 > 後端開發 > Django ORM:最全面的資料庫處理指南

Django ORM:最全面的資料庫處理指南

2023-07-11 07:45:47 後端開發

深度探討Django ORM的概念、基礎使用、進階操作以及詳細決議在實際使用中如何處理資料庫操作,同時,我們還討論了模型深入理解,如何進行CRUD操作,并且深化理解到資料庫遷移等高級主題,為了全面解讀Django ORM,我們也討論了其存在的不足,并對其未來發展進行了展望,這篇文章旨在幫助讀者全面掌握Django ORM,理解其如何簡化資料庫操作,并透過表象理解其內部作業原理,

Django ORM簡介

在深入討論Django的ORM(Object-Relational Mapping,物件-關系映射)之前,讓我們先理解一下什么是ORM,

ORM是一種編程技術,用于在面向物件的軟體和關系資料庫之間建立一種可兼容的系統,簡單來說,ORM能夠讓你使用Python(或其他編程語言)來操作資料庫,就像你在操作Python物件一樣,

Django的ORM是一個非常強大的工具,它幫助你管理和查詢資料庫,基于Django ORM的主要優勢,你可以:

  • 利用Python的物件模型進行資料庫查詢,無需撰寫復雜的SQL陳述句,
  • 實作資料庫的平臺獨立性,因為Django ORM可以在多種資料庫系統上運行,

下面我們用一個簡單的例子來說明這個概念,假設我們有一個名為"Blog"的模型,其中有一個名為"title"的欄位,使用Django ORM,我們可以輕松地查詢所有標題包含"Django"的博客,

# 匯入模型
from myapp.models import Blog

# 使用ORM進行查詢
blogs = Blog.objects.filter(title__contains='Django')

# 輸出查詢結果
for blog in blogs:
    print(blog.title)

如果你在資料庫中有名為"Learning Django"和"Django ORM basics"的博客,上面的代碼將會輸出:

Learning Django
Django ORM basics

看到這里,你可能會發現Django ORM的強大之處:它把復雜的資料庫操作轉化為Python物件操作,這極大地提高了我們的編程效率,


Django ORM運行機理與模型介紹

Django ORM運行機理

Django ORM將類(class)映射到資料庫表(table),將類的實體(instance)映射到表的記錄(record),將類的欄位(field)映射到資料庫的欄位(column),通過這種方式,你可以使用Python代碼對資料庫進行操作,而無需寫任何SQL陳述句,

在Django ORM中,每個模型(model)對應一個資料庫表,模型的欄位對應表的列,模型的實體對應表的行,

Django模型介紹

在Django中,模型是對資料庫表的一種高級抽象,通過定義一個模型,你可以明確地指定資料庫的結構,包括資料表的名稱、欄位的名稱和型別,以及可能的索引等,

讓我們看一個簡單的例子,定義一個名為“Blog”的模型:

from django.db import models

class Blog(models.Model):
    title = models.CharField(max_length=200)
    content = models.TextField()
    pub_date = models.DateTimeField(auto_now_add=True)

在這個例子中,我們定義了一個名為Blog的模型,它有三個欄位:titlecontentpub_date,每個欄位都對應一種資料庫的列型別:CharField對應字符型別,TextField對應文本型別,DateTimeField對應日期時間型別,


Django模型Model深入理解

定義模型

在Django中,模型是資料訪問層的核心組成部分,它為你的資料定義了最重要的行為,模型是一個Python類,子類于django.db.models.Model,每個模型都對應一個資料庫表,

from django.db import models

class Blog(models.Model):
    title = models.CharField(max_length=200)
    content = models.TextField()
    pub_date = models.DateTimeField(auto_now_add=True)

上面的例子中,我們定義了一個名為Blog的模型,它有三個欄位:titlecontentpub_date

模型欄位型別

Django提供了許多內置的欄位型別,可以滿足大部分的資料庫設計需求,例如:

  • CharField:字符欄位,用于存盤較短的字串,如標題,
  • TextField:文本欄位,用于存盤大量文本,如博客內容,
  • DateTimeField:日期時間欄位,用于存盤日期和時間,

每種欄位型別都有其特定的引數,例如CharField需要一個max_length引數,指定該欄位的最大長度,

模型關聯關系

Django的模型還可以定義復雜的關聯關系,包括一對一(OneToOne)、一對多(ForeignKey)和多對多(ManyToMany)關系,

例如,我們可以定義一個Author模型,并將其與Blog模型關聯:

class Author(models.Model):
    name = models.CharField(max_length=100)

class Blog(models.Model):
    title = models.CharField(max_length=200)
    content = models.TextField()
    pub_date = models.DateTimeField(auto_now_add=True)
    author = models.ForeignKey(Author, on_delete=models.CASCADE)

這里我們為Blog模型添加了一個author欄位,它是一個外鍵欄位(ForeignKey),指向Author模型,這意味著每篇博客都有一個作者,而每個作者可以寫多篇博客,

當我們洗掉一個作者時,on_delete=models.CASCADE引數將確保所有關聯的博客也會被洗掉,


Django ORM增刪改查CRUD操作

在了解了Django模型后,我們來看看如何使用Django ORM進行常見的資料庫操作:創建(Create)、讀取(Retrieve)、更新(Update)和洗掉(Delete),通常被稱為CRUD操作,

創建記錄

在Django ORM中,我們可以通過創建模型的實體來創建新的記錄,以下是一個創建新的Blog記錄的示例:

from myapp.models import Blog

# 創建新的Blog實體
blog = Blog(title='My first blog', content='This is my first blog post.')
blog.save()  # don't forget to call save method

save()方法會將新的Blog實體保存到資料庫中,

讀取記錄

Django ORM提供了多種方法來讀取資料庫中的記錄,我們可以使用all()方法獲取所有記錄,也可以使用filter()方法獲取滿足特定條件的記錄,

from myapp.models import Blog

# 獲取所有Blog記錄
blogs = Blog.objects.all()

# 輸出所有Blog的標題
for blog in blogs:
    print(blog.title)

如果有一個名為"My first blog"的博客,上面的代碼將會輸出:

My first blog

更新記錄

更新資料庫中的記錄也很簡單,我們可以獲取一個記錄的實體,修改它的屬性,然后呼叫save()方法:

from myapp.models import Blog

# 獲取第一個Blog記錄
blog = Blog.objects.first()

# 更新標題
blog.title = 'My updated blog'
blog.save()

這段代碼將更新資料庫中第一個博客的標題,

洗掉記錄

要洗掉資料庫中的記錄,我們可以獲取一個記錄的實體,然后呼叫delete()方法:

from myapp.models import Blog

# 獲取第一個Blog記錄
blog = Blog.objects.first()

# 洗掉記錄
blog.delete()

這段代碼將洗掉資料庫中的第一個博客,


Django ORM資料庫進階操作

在掌握了Django ORM的基礎操作后,接下來我們來看看一些更高級的資料庫操作,包括復雜查詢、查詢優化等,

復雜查詢

在Django ORM中,我們可以使用filter()exclude()order_by()等方法進行復雜查詢,

例如,我們可以找到所有標題包含"django"的博客,并按照發布日期降序排序:

from myapp.models import Blog

# 獲取所有標題包含'django'的Blog記錄,并按照發布日期降序排序
blogs = Blog.objects.filter(title__contains='django').order_by('-pub_date')

# 輸出這些Blog的標題
for blog in blogs:
    print(blog.title)

如果存在標題包含'django'的博客,上述代碼將會按照發布日期的降序列印它們的標題,

查詢優化

對于大型資料庫,優化查詢是非常重要的,Django ORM提供了幾種工具來幫助你優化查詢,包括select_related()prefetch_related()

select_related()可以一次性獲取與查詢物件有ForeignKey關聯的物件,這可以減少資料庫查詢次數:

from myapp.models import Blog

# 獲取所有Blog記錄,并一次性獲取每個Blog的author資訊
blogs = Blog.objects.select_related('author').all()

# 輸出Blog的標題和作者名
for blog in blogs:
    print(blog.title, blog.author.name)

如果有作者名為'John'且標題為"My first blog"的博客,上述代碼將會輸出:

My first blog John

prefetch_related()對于ManyToMany關聯和一對多關聯也非常有用,它可以一次性獲取所有相關物件,減少資料庫查詢次數,


利用資料庫約束保證資料一致性

Django ORM提供了多種資料庫約束,如uniquecheck等,可以幫助我們確保資料庫的資料一致性,

# 例子:使用unique約束確保每個作者的email是唯一的
class Author(models.Model):
    name = models.CharField(max_length=100)
    email = models.EmailField(unique=True)

使用批量操作提升性能

Django ORM提供了bulk_createbulk_update等方法,可以讓我們以更高效的方式進行批量創建和更新,

# 例子:使用bulk_create方法批量創建物件
authors = [Author(name=f'Author {i}') for i in range(1000)]
Author.objects.bulk_create(authors)  # 這個操作只需要一次資料庫查詢

使用原生SQL

盡管Django ORM提供了許多強大的查詢工具,但有時候你可能需要直接執行SQL陳述句,Django ORM允許你執行原生SQL,你可以使用raw()方法或者cursor()方法來執行原生SQL:

from django.db import connection

# 使用cursor()執行原生SQL
with connection.cursor() as cursor:
    cursor.execute("SELECT title FROM myapp_blog")
    row = cursor.fetchone()

print(row)

這段代碼將直接執行SQL查詢,并列印出第一個博客的標題,雖然Django ORM提供了.raw()方法允許我們直接執行SQL查詢,但是這個方法應該盡量避免使用,因為它可能會引發SQL注入等安全問題,同時也失去了Django ORM的許多優點,


Django資料庫遷移

Django的資料庫遷移系統能夠自動地將你對模型的更改(增加欄位、洗掉模型等)轉換為SQL命令,并在資料庫中執行這些命令,這是一種對資料庫進行版本控制的方式,讓你能夠更改你的模型并保持資料庫的更新,

創建遷移

當你更改了你的模型(例如,添加一個欄位、改變一個欄位的型別等),你需要創建一個遷移來將這些更改應用到資料庫,你可以使用makemigrations命令來創建遷移:

python manage.py makemigrations your_app_name

上述命令將會檢查你的模型與資料庫的當前狀態,然后創建一個新的遷移,該遷移包含了將資料庫更新至新狀態所需的所有操作,

應用遷移

創建了遷移之后,你需要使用migrate命令來應用這些更改到資料庫:

python manage.py migrate

該命令將執行所有未應用的遷移,將你的資料庫更新至最新狀態,

查看遷移狀態

你可以使用showmigrations命令查看所有遷移的狀態(已應用的和未應用的):

python manage.py showmigrations

執行上述命令將會列出所有的遷移以及它們的狀態,這可以幫助你了解資料庫的當前狀態,

回滾遷移

有時,你可能需要撤銷某個遷移,你可以使用migrate命令和遷移名(或遷移名之前的遷移名)來回滾遷移:

python manage.py migrate your_app_name 0001

該命令將會撤銷名為0002的遷移(以及在其之后的所有遷移),并將資料庫回滾至0001的狀態,


Django ORM的不足點

盡管Django ORM是一個強大且方便的工具,但它并不是無懈可擊的,了解這些局限性可以幫助我們更加理智地決定何時以及如何使用它,

性能開銷

Django ORM需要額外的處理來將資料庫的行轉換為Python物件,這意味著使用Django ORM通常會比直接使用SQL陳述句慢一些,然而,這種性能開銷通常是可以接受的,除非你正在處理極大量的資料,

不支持某些復雜的SQL查詢

雖然Django ORM支持許多SQL功能,但有一些復雜的SQL查詢可能無法通過Django ORM的查詢API來實作,例如,對于某些資料庫的特定特性或高級SQL功能,可能需要寫原生的SQL陳述句,

需要額外的學習和理解

雖然Django ORM可以幫助我們避免直接撰寫SQL,但是要有效地使用它,仍然需要理解資料庫的基本概念,而且,Django ORM自身的API和特性也需要一些學習和理解,

對資料庫的隱藏可能導致誤解

Django ORM隱藏了資料庫的許多細節,這使得編程變得更簡單,但也可能導致開發者對正在執行的資料庫操作有誤解,例如,一個看似簡單的操作可能實際上引發了多次資料庫查詢,

# 例子:看似簡單的操作實際上引發了多次資料庫查詢
for book in Book.objects.all():
    print(book.author.name)  # 這個操作對每本書都會引發一個資料庫查詢

在這個例子中,列印每本書的作者名字的操作實際上對每本書都會引發一個資料庫查詢,如果有大量的書籍,那么這個操作將會非常慢,這是因為Django ORM默認是懶加載的,也就是說,它只在需要的時候才會去資料庫查詢資料,


Django ORM總結與展望

經過這篇文章的學習,我們深入了解了Django ORM的作業原理,實作了一些基礎和進階的資料庫操作,同時也了解了它的一些最佳實踐和局限性,

Django ORM作為Python Web開發中的一個重要部分,它以其簡潔和強大的功能贏得了許多開發者的喜愛,盡管它有一些局限性,比如某些復雜查詢的支持不是很好,以及它對資料庫操作的隱藏可能會導致性能問題,但是總體來說,Django ORM是一個非常有效的工具,可以幫助我們更快更好地進行Web開發,

在未來,我們可以期待Django ORM將會持續改進,提供更多的功能和更好的性能,同時,我們也可以通過深入學習和實踐,更好地利用Django ORM,提高我們的開發效率,

如有幫助,請多關注
個人微信公眾號:【Python全視角】
TeahLead_KrisChang,10+年的互聯網和人工智能從業經驗,10年+技術和業務團隊管理經驗,同濟軟體工程本科,復旦工程管理碩士,阿里云認證云服務資深架構師,上億營收AI產品業務負責人,

轉載請註明出處,本文鏈接:https://www.uj5u.com/houduan/556916.html

標籤:其他

上一篇:聊一聊Java中的Steam流

下一篇:返回列表

標籤雲
其他(162306) Python(38273) JavaScript(25528) Java(18294) C(15239) 區塊鏈(8275) C#(7972) AI(7469) 爪哇(7425) MySQL(7292) html(6777) 基礎類(6313) sql(6102) 熊猫(6058) PHP(5876) 数组(5741) R(5409) Linux(5347) 反应(5209) 腳本語言(PerlPython)(5129) 非技術區(4971) Android(4615) 数据框(4311) css(4259) 节点.js(4032) C語言(3288) json(3245) 列表(3129) 扑(3119) C++語言(3117) 安卓(2998) 打字稿(2995) VBA(2789) Java相關(2746) 疑難問題(2699) 细绳(2522) 單片機工控(2479) iOS(2438) ASP.NET(2404) MongoDB(2323) 麻木的(2285) 正则表达式(2254) 字典(2211) 循环(2198) 迅速(2185) 擅长(2169) 镖(2155) HtmlCss(1994) .NET技术(1986) 功能(1967) Web開發(1951) C++(1942) python-3.x(1918) 弹簧靴(1913) xml(1889) PostgreSQL(1882) .NETCore(1863) 谷歌表格(1846) Unity3D(1843) for循环(1842)

熱門瀏覽
  • 【C++】Microsoft C++、C 和匯編程式檔案

    ......

    uj5u.com 2020-09-10 00:57:23 more
  • 例外宣告

    相比于斷言適用于排除邏輯上不可能存在的狀態,例外通常是用于邏輯上可能發生的錯誤。 例外宣告 Item 1:當函式不可能拋出例外或不能接受拋出例外時,使用noexcept 理由 如果不打算拋出例外的話,程式就會認為無法處理這種錯誤,并且應當盡早終止,如此可以有效地阻止例外的傳播與擴散。 示例 //不可 ......

    uj5u.com 2020-09-10 00:57:27 more
  • Codeforces 1400E Clear the Multiset(貪心 + 分治)

    鏈接:https://codeforces.com/problemset/problem/1400/E 來源:Codeforces 思路:給你一個陣列,現在你可以進行兩種操作,操作1:將一段沒有 0 的區間進行減一的操作,操作2:將 i 位置上的元素歸零。最終問:將這個陣列的全部元素歸零后操作的最少 ......

    uj5u.com 2020-09-10 00:57:30 more
  • UVA11610 【Reverse Prime】

    本人看到此題沒有翻譯,就附帶了一個自己的翻譯版本 思考 這一題,它的第一個要求是找出所有 $7$ 位反向質數及其質因數的個數。 我們應該需要質數篩篩選1~$10^{7}$的所有數,這里就不慢慢介紹了。但是,重讀題,我們突然發現反向質數都是 $7$ 位,而將它反過來后的數字卻是 $6$ 位數,這就說明 ......

    uj5u.com 2020-09-10 00:57:36 more
  • 統計區間素數數量

    1 #pragma GCC optimize(2) 2 #include <bits/stdc++.h> 3 using namespace std; 4 bool isprime[1000000010]; 5 vector<int> prime; 6 inline int getlist(int ......

    uj5u.com 2020-09-10 00:57:47 more
  • C/C++編程筆記:C++中的 const 變數詳解,教你正確認識const用法

    1、C中的const 1、區域const變數存放在堆疊區中,會分配記憶體(也就是說可以通過地址間接修改變數的值)。測驗代碼如下: 運行結果: 2、全域const變數存放在只讀資料段(不能通過地址修改,會發生寫入錯誤), 默認為外部聯編,可以給其他源檔案使用(需要用extern關鍵字修飾) 運行結果: ......

    uj5u.com 2020-09-10 00:58:04 more
  • 【C++犯錯記錄】VS2019 MFC添加資源不懂如何修改資源宏ID

    1. 首先在資源視圖中,添加資源 2. 點擊新添加的資源,復制自動生成的ID 3. 在解決方案資源管理器中找到Resource.h檔案,編輯,使用整個專案搜索和替換的方式快速替換 宏宣告 4. Ctrl+Shift+F 全域搜索,點擊查找全部,然后逐個替換 5. 為什么使用搜索替換而不使用屬性視窗直 ......

    uj5u.com 2020-09-10 00:59:11 more
  • 【C++犯錯記錄】VS2019 MFC不懂的批量添加資源

    1. 打開資源頭檔案Resource.h,在其中預先定義好宏 ID(不清楚其實ID值應該設定多少,可以先新建一個相同的資源項,再在這個資源的ID值的基礎上遞增即可) 2. 在資源視圖中選中專案資源,按F7編輯資源檔案,按 ID 型別 相對路徑的形式添加 資源。(別忘了先把檔案拷貝到專案中的res檔案 ......

    uj5u.com 2020-09-10 01:00:19 more
  • C/C++編程筆記:關于C++的參考型別,專供新手入門使用

    今天要講的是C++中我最喜歡的一個用法——參考,也叫別名。 參考就是給一個變數名取一個變數名,方便我們間接地使用這個變數。我們可以給一個變數創建N個參考,這N + 1個變數共享了同一塊記憶體區域。(參考型別的變數會占用記憶體空間,占用的記憶體空間的大小和指標型別的大小是相同的。雖然參考是一個物件的別名,但 ......

    uj5u.com 2020-09-10 01:00:22 more
  • 【C/C++編程筆記】從頭開始學習C ++:初學者完整指南

    眾所周知,C ++的學習曲線陡峭,但是花時間學習這種語言將為您的職業帶來奇跡,并使您與其他開發人員區分開。您會更輕松地學習新語言,形成真正的解決問題的技能,并在編程的基礎上打下堅實的基礎。 C ++將幫助您養成良好的編程習慣(即清晰一致的編碼風格,在撰寫代碼時注釋代碼,并限制類內部的可見性),并且由 ......

    uj5u.com 2020-09-10 01:00:41 more
最新发布
  • Django ORM:最全面的資料庫處理指南

    **深度探討Django ORM的概念、基礎使用、進階操作以及詳細決議在實際使用中如何處理資料庫操作。同時,我們還討論了模型深入理解,如何進行CRUD操作,并且深化理解到資料庫遷移等高級主題。為了全面解讀Django ORM,我們也討論了其存在的不足,并對其未來發展進行了展望。這篇文章旨在幫助讀者全 ......

    uj5u.com 2023-07-11 07:45:47 more
  • 聊一聊Java中的Steam流

    在我們的日常編程任務中,對于集合的制造和處理是必不可少的。當我們需要對于集合進行分組或查找的操作時,需要用迭代器對于集合進行操作,而當我們需要處理的資料量很大的時候,為了提高性能,就需要使用到并行處理,這樣的處理方式是很復雜的。流可以幫助開發者節約寶貴的時間,讓以上的事情變得輕松。 ......

    uj5u.com 2023-07-11 07:45:43 more
  • 快試試用 API Key 來保護你的 SpringBoot 介面安全吧!

    來源:baeldung.com/spring-boot-api-key-secret ## 1、概述 安全性在REST API開發中扮演著重要的角色。一個不安全的REST API可以直接訪問到后臺系統中的敏感資料。因此,企業組織需要關注API安全性。 Spring Security 提供了各種機制來 ......

    uj5u.com 2023-07-11 07:45:37 more
  • [滲透測驗]—2.2 常見的攻擊型別

    在本節中,我們將介紹一些常見的攻擊型別,以幫助你更好地了解滲透測驗的目標和方法。我們將涵蓋以下攻擊型別: 1. SQL注入攻擊 2. 跨站腳本攻擊(XSS) 3. 跨站請求偽造(CSRF) 4. 會話劫持 5. 社會工程攻擊 6. 暴力破解 7. 分布式拒絕服務攻擊(DDoS) 8. 零日攻擊 ## ......

    uj5u.com 2023-07-11 07:45:30 more
  • Go優雅的錯誤處理: 支持錯誤堆疊, 錯誤碼, 錯誤鏈的工具庫

    地址: https://github.com/morrisxyang/errors 如果覺得有用歡迎 Star 和 PR, 有問題請直接提issue # errors [![Go Reference](https://pkg.go.dev/badge/github.com/morrisxyang/e ......

    uj5u.com 2023-07-11 07:45:24 more
  • 基于JavaFX的掃雷游戲實作(四)——排行榜

    這期看標題已經能猜到了,主要講的是成績排行功能,還有對應的檔案讀寫。那么廢話不多說,讓我們有請今天的主角...的設計稿: 那么主角是何方神圣呢?當然是圖中的大框框——TableView。關于這個控制元件的選取沒有太多講究,你也可以用文本域,手動換行來顯示。我只是覺得使用表格顯示看起來更規范些。接下來考慮 ......

    uj5u.com 2023-07-11 07:40:11 more
  • Dax函式教程_編程入門自學教程_菜鳥教程-免費教程分享

    ## 教程簡介 DAX代表 Data Analysis Expressions. DAX是一種公式語言,是函式,運算子和常量的集合,可以在公式或運算式中用于計算和回傳一個或多個值. DAX是與Microsoft Excel Power Pivot和Microsoft Power BI的資料模型相關聯 ......

    uj5u.com 2023-07-11 07:40:05 more
  • python筆記:第六章函式&方法

    # 1.系統函式 由系統提供,直接拿來用或是匯入模塊后使用 ``` a = 1.12386 result = round(a,2) print(result) > 1.12 ``` # 2.自定義函式 * 函式是結構化編程的核心 * 使用關鍵詞`def`來定義函式 ``` #函式定義 def fun ......

    uj5u.com 2023-07-11 07:39:00 more
  • Go優雅的錯誤處理: 支持錯誤堆疊, 錯誤碼, 錯誤鏈的工具庫

    地址: https://github.com/morrisxyang/errors 如果覺得有用歡迎 Star 和 PR, 有問題請直接提issue # errors [![Go Reference](https://pkg.go.dev/badge/github.com/morrisxyang/e ......

    uj5u.com 2023-07-11 07:10:29 more
  • 【調制解調】AM 調幅

    ## 說明 學習數字信號處理演算法時整理的學習筆記。同系列文章目錄可見 [《DSP 學習之路》目錄](https://www.cnblogs.com/young520/p/17539849.html)。本篇介紹 AM 調幅信號的調制與解調,內附全套 MATLAB 代碼。 [TOC] ## 1. AM ......

    uj5u.com 2023-07-10 08:31:33 more