Django框架(六):模型(二) 欄位查詢、查詢集

来源:https://www.cnblogs.com/liuhui0308/archive/2020/01/19/12207595.html

1. 欄位查詢 通過模型類.objects屬性可以調用如下函數,實現對模型類對應的數據表的查詢。 函數名 功能 返回值 說明 get 返回表中滿足條件的一條且只能有一條數據。 返回值是一個模型類對象。 參數中寫查詢條件。 1)如果查到多條數據,則拋異常MultipleObjectsReturned。 ...


1. 欄位查詢

通過模型類.objects屬性可以調用如下函數,實現對模型類對應的數據表的查詢。

函數名

功能

返回值

說明

get

返回表中滿足條件的一條且只能有一條數據。

返回值是一個模型類對象。

參數中寫查詢條件。

1)如果查到多條數據,則拋異常MultipleObjectsReturned。

2)查詢不到數據,則拋異常:DoesNotExist。

all

返回模型類對應表格中的所有數據。

返回值是QuerySet類型

查詢集

filter

返回滿足條件的數據。

返回值是QuerySet類型

參數寫查詢條件。

exclude

返回不滿足條件的數據。

返回值是QuerySet類型

參數寫查詢條件。

order_by

對查詢結果進行排序。

返回值是QuerySet類型

參數中寫根據哪些欄位進行排序。

get、all前面已經使用過了,下麵主要講解filter、exclude和order_by的使用。

準備數據:

index.html:

<html>
<head>
    <title>圖書列表</title>
</head>
<body>
<h1>{{title}}</h1>
{%for i in list%}
{{i.id}}<br>
{{i.btitle}}<br>
{{i.bpub_date}}<br>
{{i.bread}}<br>
{{i.bcomment}}<br>
{%endfor%}
</body>
</html>

我們後面的查詢只需要在views.py中修改代碼就行了。

1.1 查看mysql資料庫日誌

為了方便查看查詢結果,我們先配置下,讓系統生成mysql日誌文件。

我使用的是mysql5.7版本,其他版本可以參考這個,但是建議是自己去百度。

查看mysql資料庫日誌可以查看對資料庫的操作記錄需要做如下配置:

首先查看是否開啟了binlog:show binary logs;

發現沒有開啟logs,那麼我們就可以開始配置了。

我們要去C:\ProgramData\MySQL\MySQL Server 5.7路徑下找到my.ini文件。

然後編輯這個文件,我們可以使用搜索功能。

修改成這樣:

# Binary Logging.
log-bin=mysql-bin
binlog-format=Row

如果不能修改,就使用管理員許可權打開。

然後重啟mysql服務。

驗證binlog是否開啟:show variables like 'log_bin'; 和 show binary logs;

binlog文件的位置:如果在修改my.ini的binlog時給的是全路徑,那麼生成的日誌文件就在指定的目錄下;如果只給一個名字,那麼生成的binlog日誌的位置為: 

如果給的全路徑配置為:

log-bin=指定路徑\mysql-bin
binlog-format=Row

那麼就會在指定路徑下生成文件。

1.2 條件運算符 

1.2.1 查詢

exact:表示判斷等於,和 = 差不多。

例:查詢編號為1的圖書。

list=BookInfo.objects.filter(id__exact=1)

可簡寫為:

list=BookInfo.objects.filter(id=1)

1.2.2 模糊查詢

contains:是否包含。

如果要包含%無需轉義,直接寫即可。

list = BookInfo.objects.filter(btitle__contains='鬥')

startswith、endswith:以指定值開頭或結尾。

例:查詢書名以'天'結尾的圖書。

list = BookInfo.objects.filter(btitle__endswith='')

以上運算符都區分大小寫,在這些運算符前加上i表示不區分大小寫,如iexact、icontains、istartswith、iendswith。

1.2.3 空查詢

isnull:是否為null。

例:查詢書名不為空的圖書。

list = BookInfo.objects.filter(btitle__isnull=False)

1.2.4 範圍查詢

in:是否包含在範圍內。

例:查詢編號為1或3或5的圖書。

list = BookInfo.objects.filter(id__in=[1, 3, 5])

1.2.5 比較查詢

gt、gte、lt、lte:大於、大於等於、小於、小於等於。

例:查詢編號大於3的圖書。

list = BookInfo.objects.filter(id__gt=3)

不等於的運算符,使用exclude()過濾器。

例:查詢編號不等於3的圖書

list = BookInfo.objects.exclude(id=3)

1.2.6 日期查詢

year、month、day、week_day、hour、minute、second:對日期時間類型的屬性進行運算。

例:查詢2009年發表的圖書。

list = BookInfo.objects.filter(bpub_date__year=2009)

例:查詢2008年12月14日後發表的圖書。

from datetime import date
...
list = BookInfo.objects.filter(bpub_date__gt=date(2008, 12, 14))

1.3 F對象

之前的查詢都是對象的屬性與常量值比較,兩個屬性怎麼比較呢? 答:使用F對象,被定義在django.db.models中。

語法如下:

F(屬性名)

例:查詢閱讀量大於等於評論量的圖書。

from django.db.models import F
...
list = BookInfo.objects.filter(bread__gte=F('bcomment'))

可以在F對象上使用算數運算。

例:查詢閱讀量大於2倍評論量的圖書。

list = BookInfo.objects.filter(bread__gt=F('bcomment') * 2)

1.4 Q對象

多個過濾器逐個調用表示邏輯與關係,同sql語句中where部分的and關鍵字。

例:查詢閱讀量大於20,並且編號小於3的圖書。

list=BookInfo.objects.filter(bread__gt=20,id__lt=3)
或
list=BookInfo.objects.filter(bread__gt=20).filter(id__lt=3)

如果需要實現邏輯或or的查詢,需要使用Q()對象結合|運算符,Q對象被義在django.db.models中。

語法如下:

Q(屬性名__運算符=值)

例:查詢閱讀量大於20的圖書,改寫為Q對象如下。

from django.db.models import Q
...
list = BookInfo.objects.filter(Q(bread__gt=20))

Q對象可以使用&、|連接,&表示邏輯與,|表示邏輯或。

例:查詢閱讀量大於20,或編號小於3的圖書,只能使用Q對象實現。

list = BookInfo.objects.filter(Q(bread__gt=20) | Q(pk__lt=3))

Q對象前可以使用~操作符,表示非not。

例:查詢編號不等於3的圖書。

list = BookInfo.objects.filter(~Q(pk=3))

1.5 聚合函數

使用aggregate()過濾器調用聚合函數。聚合函數包括:Avg,Count,Max,Min,Sum,被定義在django.db.models中。

例:查詢圖書的總閱讀量。

from django.db.models import Sum
...
sum = BookInfo.objects.aggregate(Sum('bread'))

修改界面

<html>
<head>
    <title>圖書列表</title>
</head>
<body>
<h1>{{title}}</h1>

{{sum}}<br>

</body>
</html>

以及

from django.shortcuts import render
from booktest.models import BookInfo
from django.db.models import Sum

def index(request):
    sum = BookInfo.objects.aggregate(Sum('bread'))

    context = {'title': '圖書列表', 'sum': sum}
    return render(request, 'booktest/index.html', context)

註意aggregate的返回值是一個字典類型,格式如下:

{'聚合類小寫__屬性名':值}
如:{'sum__bread':3}

使用count時一般不使用aggregate()過濾器。

例:查詢圖書總數。

count = BookInfo.objects.count()

參照上面修改代碼

註意count函數的返回值是一個數字。

2. 查詢集

查詢集表示從資料庫中獲取的對象集合,在管理器上調用某些過濾器方法會返回查詢集,查詢集可以含有零個、一個或多個過濾器。過濾器基於所給的參數限制查詢的結果,從Sql的角度,查詢集和select語句等價,過濾器像where和limit子句。

返回查詢集的過濾器如下:

  • all():返回所有數據。
  • filter():返回滿足條件的數據。
  • exclude():返回滿足條件之外的數據,相當於sql語句中where部分的not關鍵字。
  • order_by():對結果進行排序。

返回單個值的過濾器如下:

  • get():返回單個滿足條件的對象
    • 如果未找到會引發"模型類.DoesNotExist"異常。
    • 如果多條被返回,會引發"模型類.MultipleObjectsReturned"異常。
  • count():返回當前查詢結果的總條數。
  • aggregate():聚合,返回一個字典。

判斷某一個查詢集中是否有數據:

  • exists():判斷查詢集中是否有數據,如果有則返回True,沒有則返回False。

2.1 兩大特點

  • 惰性執行:創建查詢集不會訪問資料庫,直到調用數據時,才會訪問資料庫,調用數據的情況包括迭代、序列化、與if合用。
  • 緩存:使用同一個查詢集,第一次使用時會發生資料庫的查詢,然後把結果緩存下來,再次使用這個查詢集時會使用緩存的數據。

示例:查詢所有,編輯booktest/views.py的index視圖,運行查看。

list=BookInfo.objects.all()

2.2 查詢集的緩存

每個查詢集都包含一個緩存來最小化對資料庫的訪問。

在新建的查詢集中,緩存為空,首次對查詢集求值時,會發生資料庫查詢,django會將查詢的結果存在查詢集的緩存中,並返回請求的結果,接下來對查詢集求值將重用緩存中的結果。

演示:運行項目shell。

python manage.py shell

情況一:如下是兩個查詢集,無法重用緩存,每次查詢都會與資料庫進行一次交互,增加了資料庫的負載。

from booktest.models import BookInfo
[book.id for book in BookInfo.objects.all()]
[book.id for book in BookInfo.objects.all()]

情況二:經過存儲後,可以重用查詢集,第二次使用緩存中的數據。

list=BookInfo.objects.all()
[book.id for book in list]
[book.id for book in list]

2.3 限制查詢集

可以對查詢集進行取下標或切片操作,等同於sql中的limit和offset子句。

註意:不支持負數索引。

對查詢集進行切片後返回一個新的查詢集,不會立即執行查詢。

如果獲取一個對象,直接使用[0],等同於[0:1].get(),但是如果沒有數據,[0]引發IndexError異常,[0:1].get()如果沒有數據引發DoesNotExist異常。

示例:獲取第1、2項,運行查看。

list=BookInfo.objects.all()[0:2]


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

更多相關文章
  • 一、java中八種基本數據類型對應的包裝類型 基本數據類型 包裝類型 byte java.lang.Byte short java.lang.Short int java.lang.Integer long java.lang.Long float java.lang.Float double ja ...
  • 開發環境: Windows操作系統 開發工具: Eclipse+Jdk+Tomcat+MYSQL資料庫 運行效果圖: 源碼及原文鏈接:http://javadao.xyz/forum.php?mod=viewthread&tid=28 ...
  • 慕課網-跳跳虎-圖解+仿寫 新手都能學懂的SpringBoot源碼課-366元 慕課網-跳跳虎-圖解+仿寫 新手都能學懂的SpringBoot源碼課-366元——全方位深入解析最新版SpringBoot源碼當下SpringBoot日漸取代SSM成為新項目首選框架,企業招聘對其要求也不斷提高。掌握Sp ...
  • 概述 ReentrantLock是一個可重入的互斥鎖,也被稱為獨占鎖。它支持公平鎖和非公平鎖兩種模式。 ReentrantLock的使用方法 下麵看一個最初級的例子: 在進入方法後,在需要加鎖的一些操作執行之前需要調用lock方法,在jdk文檔中對lock方法詳細解釋如下: 獲得鎖。 如果鎖沒有被另 ...
  • 相信有不少朋友日常工作會用到 Excel 處理各式表格文件,更有甚者可能要花大把時間來做繁瑣耗時的表格整理工作。最近有朋友問可否編程來減輕表格整理工作量,今兒我們就通過實例來實現 Python 對錶格的自動化整理。 首先我們有這麼一份數據表 source.csv: 我們要做的是從上表中提取數據,來生 ...
  • SpringMVC SpringMVC是一種輕量級的、基於MVC的Web層應用框架。 通過一套 MVC 註解,讓 POJO 成為處理請求的控制器,而無須實現任何介面。 採用了鬆散耦合可插拔組件結構,比其他 MVC 框架更具擴展性和靈活性。 優點: 1、天生與Spring框架集成,如:(IOC,AOP ...
  • 本文是針對我的工具 "藍奏雲批量下載工具" 的補充說明筆記,準備按照流程整理我實現軟體的思路與方法。 涉及知識 Java的IO流 Java的下載文件 HtmlUnit的使用方法 okhttp的使用 分析與軟體思路 在某一天,我找到了一部電子書的資源,但是,該藍奏雲地址是一個文件夾,由於藍奏雲不支持批 ...
  • Java 泛型通配符 ? extends super 的用法 示例 1 : ? extends ArrayList heroList 表示這是一個Hero泛型或者其子類泛型 heroList 的泛型可能是Hero heroList 的泛型可能是APHero heroList 的泛型可能是ADHero ...
一周排行
  • 《ASP.NET MVC 企業級實戰》 [作者] (中) 鄒瓊俊[出版] 清華大學出版社[版次] 2017年04月 第1版[印次] 2019年08月 第6次 印刷[定價] 89.00元 【第01章】 (P021) 只有在 Lambda 有一個輸入參數時,括弧才是可選的,否則括弧是必需的。 使用空括弧 ...
  • 上一篇(https://www.cnblogs.com/meowv/p/12971041.html)使用HtmlAgilityPack抓取壁紙數據成功將圖片存入資料庫,本篇繼續來完成一個全網各大平臺的熱點新聞數據的抓取。 同樣的,可以先預覽一下我個人博客中的成品:https://meowv.com/ ...
  • 前言 請了一天假後回公司,同事跟我說使用Newtonsoft.json序列化TreeView對象的時候出現報錯; 啊!什麼?這個類庫不是能夠序列化所有東西嗎?真的很懵逼,也是我第一次使用這個類庫出現問題! 問題異常 異常信息 : Newtonsoft.Json.JsonSerializationEx ...
  • 簡單瞭解下麵詞語的意思 節點:二叉樹中每個元素都稱為節點 葉子節點(簡稱:葉子):度為0的節點,葉子節點就是樹中最底段的節點,葉子節點沒有子節點,也叫終端結點 分枝節點:度不為0的結點 節點的度:二叉樹的度代表某個節點的孩子或者說直接後繼的個數,簡單說就是一個節點擁有的子樹數 樹的度: 樹中最大的結 ...
  • C# 中的LINQ 提供了兩種操作方式,查詢表達式和查詢操作符,所有的查詢表達式都有對應的查操作符類替代,查詢表達式有點“類” SQL,在代碼中寫SQL,總覺得不夠“優雅”,使用查詢操作符就顯得“優雅”很多, 本系列就來對所有的LINQ 標準操作符進行一個全面的總結,這些操作符和我上篇文章總結的Rx ...
  • 在Startup ConfigureServices 註冊本地化所需要的服務AddLocalization和 Configure<RequestLocalizationOptions> public void ConfigureServices(IServiceCollection services ...
  • 為什麼需要持久化,以及Redis持久化的RDB方式在這篇文章講的已經很透徹了,足以弔打面試官了。而且此篇內容需要RDB文章的內容支持,所以建議先看下:看完這篇還不懂Redis的RDB持久化,你們來打我! 一、什麼是AOF 它也是Redis持久化的重要手段之一,aof->Append Only Fil ...
  • 先上圖: @IT程式猿 微博網友評論: @迢書:前同事的,親眼見過 @AvenGeeker:Bug 404 @科技州:這是要逼死強迫症 @小島一瞥:哈哈哈哈哈我老家的車 最後小編整理了一套技術資料不僅能精準消除技術盲點、累計面試經驗,更可以攻剋JVM、Spring、分散式、微服務等技術難題。 海量電 ...
  • 概括來說,分三步: 1,首先找到是哪個進程的CPU占有率飆到了100%。 2,根據進程號pid,定位到是哪個線程,找到對應線程的tid。 3,導出對應線程的dump日誌文件,分析日誌文件定位具體代碼。 要解決這個問題,你應該具備以下技能: 1,linux的top命令。 2,jvm監控工具jps。 3 ...
  • 寫在最後 程式員為何害怕【別人的代碼】呢?這讓我想起一個段子。 寫這段代碼時 只有上帝和我知道他是幹嘛的 現在 只有上帝知道了 別人的代碼,似乎總意味著冗長、晦澀、凌亂,給人一種不想靠近的感覺。搞笑的是,對於一些程式員而言,即使是自己的代碼,在一段時間之後自己再拿來看,也成了【別人的代碼】... 作 ...