Django1.8教程——從零開始搭建一個完整django博客(三)

来源:http://www.cnblogs.com/Lands-ljk/archive/2016/07/31/5723828.html
-Advertisement-
Play Games

這一節主要介紹對資料庫的訪問操作:通過管理器(manage),對對象進行檢索、修改、刪除等操作,詳細介紹瞭如何針對不同的模型自定義管理器。 查詢和管理工作 現在,我們已經有了一個功能完善的Django管理站點來管理我們的應用,是時候學習一下如何從資料庫中檢索我們所需要的數據。Django自帶一個功能 ...


這一節主要介紹對資料庫的訪問操作:通過管理器(manage),對對象進行檢索、修改、刪除等操作,詳細介紹瞭如何針對不同的模型自定義管理器。

查詢和管理工作

現在,我們已經有了一個功能完善的Django管理站點來管理我們的應用,是時候學習一下如何從資料庫中檢索我們所需要的數據。Django自帶一個功能強大的抽象資料庫API,使我們能輕鬆的創建、檢索、修改、和刪除對象。Django的對象關係映射(ORM)相容MySQL,PostgreSQL,SQLite,Oracle。

記住,你可以在settings.py文件中自定義你想使用的資料庫。Django可以同時運行複數資料庫,甚至可以用你喜歡的方式來操作遠程資料庫。

一旦你已經創建了你的數據模型,就可以使用API來操作它們。想要知道更多有關於數據模型的信息,可以訪問:https://docs.djangoproject.com/en/1.8/ref/models/.

創建對象

打開終端輸入以下命令來打開交互界面:

python manage.py shell

輸入以下內容:

>>>  from django.contrib.auth.models import User
>>>  from blog.models import Post
>>>  user = User.objects.get(username='admin')
>>>  Post.objects.create(title='One more post', slug='one-more-post', body='Post body.', author=user) 
>>>  post.save ()

讓我們分析一下上面的代碼:

首先,我們檢索用戶名為admin的User對象。

user = User.objects.get(username='admin')

get()方法允許你從資料庫中檢索單個對象,這個方法返回一個匹配的對象,如果沒有檢索到該對象,這個方法將會拋出DoesNotExist(對象不存在)的錯誤。如果有多個結果匹配,則會拋出一個MultipleObjectsReturned(返回覆數對象)的錯誤,這兩個異常都是當前正在查詢的模型類的屬性。

接著,我們創建一個自定義的標題(title)、簡短標題(slug)、文章內容(body)、還有設置我們剛纔檢索出來的user對象,作為文章的作者(Author)

post = Post(title='Another post', slug='another-post', body='Post body.', author=user)

然後,我們用save()方法來保存剛剛創建的Post對象。

post.save()

這一動作將會在SQL後臺註入Post對象,我們已經知道如何在記憶體中創建一個Post對象並把它保存到資料庫中。當然,我們也可以用create()方法來直接在資料庫中創建一個Post對象。

Post.objects.create(title='One more post', slug='one-more-post', body='Post body.', author=user)

更新對象

現在,我們改變一下剛剛創建的Post對象的標題並再次保存它:

>>>  post.title = 'New title'
>>>  post.save ()

這次,save()方法執行了更新資料庫(UPDATE SQL)語句

註意:直到你執行save()方法,你對對象的更改才會保存到資料庫中。

檢索對象

Django的對象關係映射(ORM)基於QuerySet(查詢集),你可以通過多種過濾方式來限制返回結果。你已經知道可以通過get()方法來從資料庫中獲取單個對象。正如你所見,可以使用Post.objects.get()來獲取指定對象。每個Django模型至少含有一個名叫objects的預設管理器。在不知不覺中,你已經通過控制器獲取了對象。同樣,你也可以使用Post.objects.all()來獲取所有對象。

>>> all_posts = Post.objects.all()

這就是如何從一個資料庫中獲取所有對象的方法:通過管理器objects要求它返回所有對象。但是要註意一點:上述語句並沒有在資料庫中立即執行,究其原因,是因為QuerySet其實非常“懶惰”,它會謹慎的評估是否需要立即執行SQL語句。而正是因為這個“懶惰”的特性,才讓它十分高效。舉個例子說吧:如果我們不是把QuerySet保存到變數(all_posts = Post.objects.all()),而是打開Python shell,輸入Post.objects.all(),SQL語句就會立即執行,是因為我們“強迫”它給我們一個交代:把查詢結果輸出到控制台.

>>> Post.objects.all()

使用管理器的filter()方法

如果我們只想查詢符合條件的對象集合,可以使用管理器的filter()方法。舉個例子,如果我們只想從所有對象中查詢2015年發佈的文章,那麼可以用:Post.objects.filter(publish__year=2015)

還有,你可以添加複數過濾條件。例如,我們想查詢作者Admin在2015年發佈的文章:

Post.objects.filter(publish__year=2015, author__username='admin')

Post.objects.filter(publish__year=2015)\
             .filter(author__username='admin')

以上兩條語句可以達到相同的查詢結果。

不知道你有沒有註意到,我們使用欄位過濾查詢結果,連著用了兩個下劃線(publish__year),而訪問相關聯模型(User)的欄位(author)時,也用了兩個下劃線(author_username)。

使用管理器的exclud()方法

有時候你想從過濾結果中剔除不合格的結果,可以使用管理器的exclud()方法,比如,我們想查詢所有2015年發佈的文章,但是剔除以“Why”開頭的文章:

Post.objects.filter(publish__year=2015)\
            .exclude(title__startswith='Why')

使用管理器的order_by()方法

如果想要讓查詢結果按照某種規則排序的時候,可以使用管理器的order_by()方法。舉個例子,如果你想讓查詢結果按照發佈日期升序排序,可以使用:(譯者註:原文是使用title作為排序方式,也就是按字母排序,譯者覺得如果是中文標題按照title排序,效果不明顯,所以改用publish排序)

Post.objects.order_by('publish')

升序規則是預設的,如果你想讓查詢結果按照發佈日期降序排序,可以這麼做:

Post.objects.order_by('-publish')

使用管理器的order_by()方法

如果你想刪除某個對象,可以用:

post = Post.objects.get(id=1)
post.delete()

請註意一點,刪除這個對象,意味著同樣刪除了與其有關的對象。(譯者註:例如,刪除了一個作者對象,則這個作者所寫的所有文章都會隨之刪除。)

資料庫何時真正執行語句

前面講過,QuerySet其實是非常“懶惰”的,你可以添加任意多的篩選條件,但是Queryset不會馬上執行,而是謹慎的對語句進行評估,當它認為有必要執行SQL語句的時候,它才會執行。那什麼情況下才是必要的?

  • 第一次遍歷查詢結果的時候
  • 當對查詢結果進行切片操作的時候:Post.objects.all()[:3]
  • 當對查詢結果進行緩存操作的時候
  • 使用repr()或者len() 的時候
  • 明確要求遍歷list()查詢結果的時候
  • 在以下測試語句中:bool(), or , and, if

創建自定義模型管理器

前面講過,objects是模型的預設管理器,每一個模型都有一個objects管理器,用於對資料庫的檢索操作。但是,我們想為我們的模型自定義一個管理器,該怎麼辦呢?比如說,我們想要創建一個獲取所有已發佈文章的管理器。

這裡有兩種方式來創建:為原始管理器添加額外的方法,或是修改預設管理器來創建新的管理器。前面一種就變成了Post.objects.my_manager(),後面一種則變成Post.my_manager.all()

我們推薦後一種方法,比如我們為Post模型創建了一個名為Published的管理器:

打開models.py文件天際一個自定義管理器:

class PublishedManager(models.Manager):
    def get_queryset(self):
        return super(PublishedManager,self).get_queryset().filter(status='published')

class Post(models.Model):
    # ...
    objects = models.Manager()   # 預設管理器.
    published = PublishedManager()    # 我們定義的管理器.

PublishedManager顯式調用父類的get_queryset()方法,返回一個包含所有對象的QuerySet,再由filter()過濾,得到一個新的QuerySet。至此,我們創建了一個新的管理器,並將它添加到Post模型中,現在我們可以用它來執行查找操作。例如,我們想查找已經發佈的文章中,標題以”who“開頭的文章。

Post.published.filter(title__startswith='Who')

原文鏈接:http://www.landsblog.com/blog/content/djangoexamplequery
更多教程:http://www.landsblog.com/blog/tag/django


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 前段時間,一直有練習ASP.NET MVC與Web API交互,接下來,Insus.NET再做一些相關的練習,Web API與文件操作,如POST文件至Web API,更新或是刪除等。不管怎樣,先在資料庫創建一張表,用來存儲上傳的文件。本實例中是把文件存儲過資料庫的。 CREATE TABLE Ap ...
  • 1.continue,break,ruturn eg:1-100的和 結果為:5050 換為break,查看結果 結果為:10 結論一:break:跳出整個迴圈體 換為continue看一下結果又是多少? 結果為:5045,(除5之外都執行) 結論二:continue跳過當前條件的迴圈 return ...
  • 目錄索引 【無私分享:ASP.NET CORE 項目實戰】目錄索引 簡介 本章我們來介紹下Asp.net Core 使用 CodeFirst 創建資料庫和表,通過 控制台 和 dotnet ef 兩種方式 修改EF上下文對象,添加測試類 我修改了一下名字,Domains 改為了 wkmvc.Data ...
  • 字元串操作大概是電腦程式中最常見的操作了,Java中表示字元串的類是String,它有哪些方法?內部是如何實現的?如何處理各種不同的編碼?不可變性意味著什麼?字元串常量到底是什麼?hashCode是如何實現的?... ...
  • lambda是一種匿名函數,python lambda可以使簡單的函數簡潔的表達,,C++的lambda使類似嵌套函數的功能得以實現 python的lambda VC++14的lambda lambda是vc++獨有的,在vc++11以後,擴展這個功能主要是為了使代碼書寫簡潔,gcc沒有這個功能 直 ...
  • 二叉查找樹,又名二叉排序樹,亦名二叉搜索樹,一種常用的樹形數據結構 ...
  • 1:為什麼會有JSP jsp全名(java server pages)中文叫做java伺服器頁面。在Servlet那一篇我們發現用Servlet可以生成動態頁面,但是我們卻在Servlet中卻寫了大量的html標簽,此外在Servlet中我們不得不將大量靜態顯示內容和動態生產內容混合在一起,使得我們 ...
  • 五、變數與賦值 Python中的變數在聲明時不需要指定其類型,它會根據你的賦值自動判斷 對於數字和字元這種值類型的賦值,變數只是對它的引用,並不能直接刪除其值(其實由於Python中的垃圾回收機制,你並不能真正刪除任何東西,由Python自動判斷它“沒有人”了才會刪除它) 六、數字 int:有符號整 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...