Django 聯(lián)表查詢操作方法
在日常的開發(fā)中,常常需要對多張數(shù)據(jù)表同時進行數(shù)據(jù)查詢。多表查詢需要在數(shù)據(jù)表之間建立表關(guān)系才能夠?qū)崿F(xiàn)。一對多或一對一的表關(guān)系是通過外鍵實現(xiàn)關(guān)聯(lián)的,而多表查詢分為正向查詢和反向查詢。
表模型結(jié)構(gòu)
以歌手表、專輯表、單曲表查詢?yōu)槔印?/p>
歌手與專輯為一對多關(guān)系;歌手和單曲為一對多關(guān)系;專輯和單曲為多對多關(guān)系;
表模型如下:
class Singler(BaseModel): """ 歌手表模型 """ name = models.CharField(max_length=50) first_letter = models.CharField(max_length=15, editable=False) # 設(shè)置上傳位置 portrait = models.ImageField(upload_to=upload_save_path) birthday = models.DateField(default=date.today,blank=True) height = models.IntegerField(default=0,blank=True) weight = models.IntegerField(default=0,blank=True) constellation = models.CharField(max_length=50) english_name = models.CharField(max_length=50,default='-') gender = models.IntegerField(choices=((0, '女'), (1, '男')),default=1) country_name = models.CharField(max_length=50,default='-') desc = models.TextField() class Singe(BaseModel): """ 單曲表 """ name = models.CharField(max_length=50) duration = models.IntegerField(editable=False, default=0) playnum = models.IntegerField(default=0, editable=False) path = models.FileField(upload_to=upload_save_path) lyric = models.FileField(upload_to=upload_save_path) # 設(shè)置與歌手表關(guān)聯(lián)外鍵 一對多外鍵設(shè)置在多的模型中 singler = models.ForeignKey("Singler",on_delete=models.CASCADE) class Album(BaseModel): """ 專輯表 """ name = models.CharField(max_length=50) cover = models.ImageField(upload_to=upload_save_path) desc = models.CharField(max_length=255) single_num = models.IntegerField(default=0,editable=False) langs = [ ('國語', '國語'), ('普通話', '普通話'), ('英語', '英語'), ('日韓', '日韓') ] single_lang = models.CharField(max_length=50,choices=langs,) # 設(shè)置與歌手表關(guān)聯(lián)外鍵 一對多 singler = models.ForeignKey("Singler",on_delete=models.CASCADE) # 設(shè)置與單曲表關(guān)聯(lián)外鍵 多對多 Singe = models.ManyToManyField('Singe')
單曲獲取歌手
通過singe模型關(guān)聯(lián)外鍵singler獲取關(guān)聯(lián)歌手Singler信息,為正向查詢。
代碼如下:
info = Singe.objects.filter(id=1).first() print('單曲信息:', info) article = info.singler print('歌手信息:', article)
效果:
歌手獲取單曲
通過歌手模型獲取單曲相應(yīng)記錄,因為外鍵在單曲表模型中,
這樣屬于反向查詢。
方法一
使用小寫模型名_set方式查詢。
代碼如下
info = Singler.objects.filter(id=1).first() print('歌手信息:', info) song = info.singe_set.first() print('一首單曲:', song) songs = info.singe_set.all() print('全部單曲:', songs)
方法二
需要對外鍵設(shè)置related_name為某個字符串,來進行關(guān)聯(lián)查詢。
模型外鍵設(shè)置
singler = models.ForeignKey("Singler", on_delete=models.CASCADE,related_name='singler_info')
視圖代碼如下:
info = Singler.objects.filter(id=1).first() print('歌手信息:', info) song = info.singler_info.first() print('一首單曲:', song) songs = info.singler_info.all() print('全部單曲:', songs)
效果:
查詢關(guān)聯(lián)條件記錄
正向查詢
通過單曲表查詢歌手名稱是周杰倫的單曲和歌手信息。
代碼如下:
info = Singe.objects.filter(singler__name='周杰倫').first() print('單曲信息:', info) article = info.singler print('歌手信息:', article)
singler是關(guān)聯(lián)外鍵,name是歌手表name字段,兩者使用雙下劃連接;
singler是Singler模型在Singe模型中設(shè)置的外鍵。
效果:
反向查詢
通過歌手表查詢歌曲名稱獲取歌手信息和單曲信息。
代碼如下:
info = Singler.objects.filter(singler_info__name='告白氣球').first() print('歌手信息:', info) song = info.singler_info.first() print('單曲信息:', song)
singler_info是models.py中表模型外鍵設(shè)置的屬性related_name='singler_info'。
通過單曲名稱獲取相應(yīng)單曲的歌手信息,
之后通過參數(shù)singler_info反向獲取模型Singe的數(shù)據(jù)。
效果:
聯(lián)表查詢優(yōu)化
無論是正向查詢還是反向查詢,它們在數(shù)據(jù)庫里需要執(zhí)行兩次SQL查詢,第一次是查詢某張數(shù)據(jù)表的數(shù)據(jù),再通過外鍵關(guān)聯(lián)獲取另一張數(shù)據(jù)表的數(shù)據(jù)信息。為了減少查詢次數(shù),提高查詢效率,我們可以使用select_related或prefetch_related方法實現(xiàn),該方法只需執(zhí)行一次SQL查詢就能實現(xiàn)多表查詢。
Select_related
select_related主要針對一對一和一對多關(guān)系進行優(yōu)化,它是使用SQL的JOIN語句進行優(yōu)化的,通過減少SQL查詢的次數(shù)來進行優(yōu)化和提高性能。
正向查詢
select_related方法,參數(shù)為字符串格式,以模型Singe為查詢對象;
select_related使用INNER JOIN方式查詢兩個數(shù)據(jù)表;
查詢模型Singe的字段singler和模型Singler的字段id;
select_related參數(shù)為singler為外鍵字段;
若要得到其他數(shù)據(jù)表的關(guān)聯(lián)數(shù)據(jù),則可用雙下畫線“__”連接字段名;
雙下畫線“__”連接字段名必須是外鍵字段名或外鍵字段related_name設(shè)置參數(shù)。
代碼如下:
p = Singe.objects.select_related('getname'). values('id', 'name', 'duration', 'singler__name') # # 查看SQL查詢語句 print(p.query) # 查看結(jié)果 為dict格式 print(p)
效果:
反向查詢
以模型Vocation為查詢對象
select_related使用LEFT JOIN方式查詢兩個數(shù)據(jù)表
select_related的參數(shù)為related_name設(shè)置參數(shù),屬于關(guān)聯(lián)表字段。
代碼如下:
f = Singler.objects.select_related('getname'). values('id', 'name', 'getname__name') # 查看SQL查詢語句 print(f.query) # 查看結(jié)果 print(f) print('#'*100) # 獲取兩個模型的數(shù)據(jù),以模型Singler的singe_num大于1為查詢條件 f = Singler.objects.select_related('getname'). filter(singe_num__gt=1).values('id', 'name', 'getname__name') # 查看SQL查詢語句 print(f.query) # 獲取查詢結(jié)果集的首個元素的字段getname__name的值 print(f[0]['getname__name'])
效果:
Prefetch_related
prefetch_related和select_related的設(shè)計目的很相似,都是為了減少SQL查詢的次數(shù),但是實現(xiàn)的方式不一樣。select_related是由SQL的JOIN語句實現(xiàn)的,但是對于多對多關(guān)系,使用select_related會增加數(shù)據(jù)查詢時間和內(nèi)存占用;而prefetch_related是分別查詢每張數(shù)據(jù)表,然后由Python語法來處理它們之間的關(guān)系,因此對于多對多關(guān)系的查詢,prefetch_related更有優(yōu)勢。
設(shè)置related_name
# 設(shè)置與單曲表關(guān)聯(lián)外鍵 多對多 Singe = models.ManyToManyField( 'Singe', verbose_name='單曲', help_text='請選擇單曲', related_name='singe_info' )
視圖處理
Album模型與Singe模型關(guān)系為多對多,也就是專輯可以添加多個單曲,單曲也可以加入多個專輯。查詢單曲名稱為告白氣球的加入了哪些專輯。
代碼如下:
s = Singe.objects.prefetch_related('singe_info').filter(name='告白氣球').first() print(s) # # 根據(jù)外鍵字段singe獲取當(dāng)前數(shù)據(jù)的多對多或一對多關(guān)系 print(s.singe_info.all()) print('#'*100) # 使用values_list獲取聯(lián)合查詢數(shù)據(jù) s = Singe.objects.prefetch_related('singe_info').filter(name='告白氣球')\ .values_list('id', 'name', 'singe_info__name') # 查看sql print(s.query) # 查看結(jié)果 print(s) # 輸出專輯名 print(s[0][2])
效果:
執(zhí)行sql語句
也可以通過raw方式,將查詢條件使用原生SQL語法實現(xiàn),
此方法需要依靠模型對象,在某程度上可防止SQL注入。
Raw查詢所有
單曲表和歌手表聯(lián)查,查詢所有數(shù)據(jù)。
代碼如下:
s1 = Singe.objects.raw('select * from player_singe as s left join player_singler as a on s.singler_id = a.id') print('查詢結(jié)果') print(s1) for item in s1: print(item)
效果:
Raw條件查詢
單曲表和歌手表聯(lián)查,查詢單曲名稱為‘告白氣球’。
代碼如下:
s = Singe.objects.raw('select * from player_singe as s left join player_singler as a on s.singler_id = a.id where s.name = "告白氣球"') print('查詢結(jié)果') print(s.query) print(s) print(s[0])
效果:
Execute查詢
execute執(zhí)行SQL語句無須經(jīng)過Django的ORM框架。借助第三方模塊實現(xiàn)連接過程,如MySQL的mysqlclient模塊和SQLite的sqlite3模塊等,這些模塊連接數(shù)據(jù)庫之后,可通過游標的方式來執(zhí)行SQL語句。很容易受到SQL注入攻擊,需要自己做參數(shù)的驗證和過濾操作。
代碼如下:
from django.db import connection cursor = connection.cursor() # 執(zhí)行SQL語句 cursor.execute('select * from player_singe as s left join player_singler as a on s.singler_id = a.id') # 讀取第一行數(shù)據(jù) print(cursor.fetchone()) # 讀取所有數(shù)據(jù) print(cursor.fetchall())
效果:
總結(jié)
作為一個django使用的新手,在做練手項目中對聯(lián)表查詢感覺比較生疏,最近兩天整理了一些連表查詢應(yīng)用場景和使用方法;以及無法使用django中ORM操作的原生查詢,以備之后忘記用作參考使用。
到此這篇關(guān)于Django 聯(lián)表查詢操作的文章就介紹到這了,更多相關(guān)Django 聯(lián)表查詢操作內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
python學(xué)習(xí)之使用Matplotlib畫實時的動態(tài)折線圖的示例代碼
這篇文章主要介紹了python學(xué)習(xí)之使用Matplotlib畫實時的動態(tài)折線圖的示例代碼,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2021-02-02python+opencv實現(xiàn)文字顏色識別與標定功能
最近小編接了一個比較簡單的圖像處理的單子,今天小編給大家分享python+opencv實現(xiàn)文字顏色識別與標定功能的完整思路及代碼,感興趣的朋友一起看看吧2021-09-09如何將numpy二維數(shù)組中的np.nan值替換為指定的值
這篇文章主要介紹了將numpy二維數(shù)組中的np.nan值替換為指定的值操作,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2021-05-05python卸載numpy出現(xiàn)WinError:拒絕訪問的解決方案
這篇文章主要介紹了python卸載numpy出現(xiàn)WinError:拒絕訪問的解決方案,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教2023-08-08