這一節主要介紹對資料庫的訪問操作:通過管理器(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