Django 聯(lián)表查詢操作方法
在日常的開(kāi)發(fā)中,常常需要對(duì)多張數(shù)據(jù)表同時(shí)進(jìn)行數(shù)據(jù)查詢。多表查詢需要在數(shù)據(jù)表之間建立表關(guān)系才能夠?qū)崿F(xiàn)。一對(duì)多或一對(duì)一的表關(guān)系是通過(guò)外鍵實(shí)現(xiàn)關(guān)聯(lián)的,而多表查詢分為正向查詢和反向查詢。
表模型結(jié)構(gòu)
以歌手表、專輯表、單曲表查詢?yōu)槔印?/p>
歌手與專輯為一對(duì)多關(guān)系;歌手和單曲為一對(duì)多關(guān)系;專輯和單曲為多對(duì)多關(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)外鍵 一對(duì)多外鍵設(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 = [
('國(guó)語(yǔ)', '國(guó)語(yǔ)'),
('普通話', '普通話'),
('英語(yǔ)', '英語(yǔ)'),
('日韓', '日韓')
]
single_lang = models.CharField(max_length=50,choices=langs,)
# 設(shè)置與歌手表關(guān)聯(lián)外鍵 一對(duì)多
singler = models.ForeignKey("Singler",on_delete=models.CASCADE)
# 設(shè)置與單曲表關(guān)聯(lián)外鍵 多對(duì)多
Singe = models.ManyToManyField('Singe')單曲獲取歌手
通過(guò)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)效果:

歌手獲取單曲
通過(guò)歌手模型獲取單曲相應(yīng)記錄,因?yàn)橥怄I在單曲表模型中,
這樣屬于反向查詢。
方法一
使用小寫模型名_set方式查詢。
代碼如下
info = Singler.objects.filter(id=1).first()
print('歌手信息:', info)
song = info.singe_set.first()
print('一首單曲:', song)
songs = info.singe_set.all()
print('全部單曲:', songs)方法二
需要對(duì)外鍵設(shè)置related_name為某個(gè)字符串,來(lái)進(jìn)行關(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)條件記錄
正向查詢
通過(guò)單曲表查詢歌手名稱是周杰倫的單曲和歌手信息。
代碼如下:
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è)置的外鍵。
效果:

反向查詢
通過(guò)歌手表查詢歌曲名稱獲取歌手信息和單曲信息。
代碼如下:
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'。
通過(guò)單曲名稱獲取相應(yīng)單曲的歌手信息,
之后通過(guò)參數(shù)singler_info反向獲取模型Singe的數(shù)據(jù)。
效果:

聯(lián)表查詢優(yōu)化
無(wú)論是正向查詢還是反向查詢,它們?cè)跀?shù)據(jù)庫(kù)里需要執(zhí)行兩次SQL查詢,第一次是查詢某張數(shù)據(jù)表的數(shù)據(jù),再通過(guò)外鍵關(guān)聯(lián)獲取另一張數(shù)據(jù)表的數(shù)據(jù)信息。為了減少查詢次數(shù),提高查詢效率,我們可以使用select_related或prefetch_related方法實(shí)現(xiàn),該方法只需執(zhí)行一次SQL查詢就能實(shí)現(xiàn)多表查詢。
Select_related
select_related主要針對(duì)一對(duì)一和一對(duì)多關(guān)系進(jìn)行優(yōu)化,它是使用SQL的JOIN語(yǔ)句進(jìn)行優(yōu)化的,通過(guò)減少SQL查詢的次數(shù)來(lái)進(jìn)行優(yōu)化和提高性能。
正向查詢
select_related方法,參數(shù)為字符串格式,以模型Singe為查詢對(duì)象;
select_related使用INNER JOIN方式查詢兩個(gè)數(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查詢語(yǔ)句
print(p.query)
# 查看結(jié)果 為dict格式
print(p)效果:

反向查詢
以模型Vocation為查詢對(duì)象
select_related使用LEFT JOIN方式查詢兩個(gè)數(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查詢語(yǔ)句
print(f.query)
# 查看結(jié)果
print(f)
print('#'*100)
# 獲取兩個(gè)模型的數(shù)據(jù),以模型Singler的singe_num大于1為查詢條件
f = Singler.objects.select_related('getname').
filter(singe_num__gt=1).values('id', 'name', 'getname__name')
# 查看SQL查詢語(yǔ)句
print(f.query)
# 獲取查詢結(jié)果集的首個(gè)元素的字段getname__name的值
print(f[0]['getname__name'])效果:

Prefetch_related
prefetch_related和select_related的設(shè)計(jì)目的很相似,都是為了減少SQL查詢的次數(shù),但是實(shí)現(xiàn)的方式不一樣。select_related是由SQL的JOIN語(yǔ)句實(shí)現(xiàn)的,但是對(duì)于多對(duì)多關(guān)系,使用select_related會(huì)增加數(shù)據(jù)查詢時(shí)間和內(nèi)存占用;而prefetch_related是分別查詢每張數(shù)據(jù)表,然后由Python語(yǔ)法來(lái)處理它們之間的關(guān)系,因此對(duì)于多對(duì)多關(guān)系的查詢,prefetch_related更有優(yōu)勢(shì)。
設(shè)置related_name
# 設(shè)置與單曲表關(guān)聯(lián)外鍵 多對(duì)多
Singe = models.ManyToManyField(
'Singe',
verbose_name='單曲',
help_text='請(qǐng)選擇單曲',
related_name='singe_info'
)視圖處理
Album模型與Singe模型關(guān)系為多對(duì)多,也就是專輯可以添加多個(gè)單曲,單曲也可以加入多個(gè)專輯。查詢單曲名稱為告白氣球的加入了哪些專輯。
代碼如下:
s = Singe.objects.prefetch_related('singe_info').filter(name='告白氣球').first()
print(s)
# # 根據(jù)外鍵字段singe獲取當(dāng)前數(shù)據(jù)的多對(duì)多或一對(duì)多關(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語(yǔ)句
也可以通過(guò)raw方式,將查詢條件使用原生SQL語(yǔ)法實(shí)現(xiàn),
此方法需要依靠模型對(duì)象,在某程度上可防止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語(yǔ)句無(wú)須經(jīng)過(guò)Django的ORM框架。借助第三方模塊實(shí)現(xiàn)連接過(guò)程,如MySQL的mysqlclient模塊和SQLite的sqlite3模塊等,這些模塊連接數(shù)據(jù)庫(kù)之后,可通過(guò)游標(biāo)的方式來(lái)執(zhí)行SQL語(yǔ)句。很容易受到SQL注入攻擊,需要自己做參數(shù)的驗(yàn)證和過(guò)濾操作。
代碼如下:
from django.db import connection
cursor = connection.cursor()
# 執(zhí)行SQL語(yǔ)句
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é)
作為一個(gè)django使用的新手,在做練手項(xiàng)目中對(duì)聯(lián)表查詢感覺(jué)比較生疏,最近兩天整理了一些連表查詢應(yīng)用場(chǎng)景和使用方法;以及無(wú)法使用django中ORM操作的原生查詢,以備之后忘記用作參考使用。
到此這篇關(guān)于Django 聯(lián)表查詢操作的文章就介紹到這了,更多相關(guān)Django 聯(lián)表查詢操作內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
python學(xué)習(xí)之使用Matplotlib畫實(shí)時(shí)的動(dòng)態(tài)折線圖的示例代碼
這篇文章主要介紹了python學(xué)習(xí)之使用Matplotlib畫實(shí)時(shí)的動(dòng)態(tài)折線圖的示例代碼,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2021-02-02
python+opencv實(shí)現(xiàn)文字顏色識(shí)別與標(biāo)定功能
最近小編接了一個(gè)比較簡(jiǎn)單的圖像處理的單子,今天小編給大家分享python+opencv實(shí)現(xiàn)文字顏色識(shí)別與標(biāo)定功能的完整思路及代碼,感興趣的朋友一起看看吧2021-09-09
如何將numpy二維數(shù)組中的np.nan值替換為指定的值
這篇文章主要介紹了將numpy二維數(shù)組中的np.nan值替換為指定的值操作,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-05-05
python卸載numpy出現(xiàn)WinError:拒絕訪問(wèn)的解決方案
這篇文章主要介紹了python卸載numpy出現(xiàn)WinError:拒絕訪問(wèn)的解決方案,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-08-08
Python后臺(tái)管理員管理前臺(tái)會(huì)員信息的講解
今天小編就為大家分享一篇關(guān)于Python后臺(tái)管理員管理前臺(tái)會(huì)員信息的講解,小編覺(jué)得內(nèi)容挺不錯(cuò)的,現(xiàn)在分享給大家,具有很好的參考價(jià)值,需要的朋友一起跟隨小編來(lái)看看吧2019-01-01

