Django 聯(lián)表查詢操作方法
在日常的開發(fā)中,常常需要對多張數(shù)據(jù)表同時進(jìn)行數(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為某個字符串,來進(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)條件記錄
正向查詢
通過單曲表查詢歌手名稱是周杰倫的單曲和歌手信息。
代碼如下:
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)系進(jìn)行優(yōu)化,它是使用SQL的JOIN語句進(jìn)行優(yōu)化的,通過減少SQL查詢的次數(shù)來進(jìn)行優(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ù)庫之后,可通過游標(biāo)的方式來執(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)折線圖的示例代碼,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2021-02-02
python+opencv實現(xiàn)文字顏色識別與標(biāo)定功能
最近小編接了一個比較簡單的圖像處理的單子,今天小編給大家分享python+opencv實現(xiàn)文字顏色識別與標(biāo)定功能的完整思路及代碼,感興趣的朋友一起看看吧2021-09-09
如何將numpy二維數(shù)組中的np.nan值替換為指定的值
這篇文章主要介紹了將numpy二維數(shù)組中的np.nan值替換為指定的值操作,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2021-05-05
python卸載numpy出現(xiàn)WinError:拒絕訪問的解決方案
這篇文章主要介紹了python卸載numpy出現(xiàn)WinError:拒絕訪問的解決方案,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教2023-08-08

