欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Django 聯(lián)表查詢操作方法

 更新時間:2023年09月27日 15:14:36   作者:JSON_L  
作為一個django使用的新手,在做練手項目中對聯(lián)表查詢感覺比較生疏,最近兩天整理了一些連表查詢應(yīng)用場景和使用方法以及無法使用django中ORM操作的原生查詢,對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測試人員需要掌握的知識

    Python測試人員需要掌握的知識

    很多朋友都想做了個python的測試人員,那么python測試員需要知道的基本知識有哪些呢?跟著小編一起學(xué)習(xí)下。
    2018-02-02
  • python學(xué)習(xí)之使用Matplotlib畫實時的動態(tài)折線圖的示例代碼

    python學(xué)習(xí)之使用Matplotlib畫實時的動態(tài)折線圖的示例代碼

    這篇文章主要介紹了python學(xué)習(xí)之使用Matplotlib畫實時的動態(tài)折線圖的示例代碼,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2021-02-02
  • 使用python遍歷指定城市的一周氣溫

    使用python遍歷指定城市的一周氣溫

    本文主要介紹了使用python遍歷指定城市的一周氣溫的實現(xiàn)方法。具有很好的參考價值,下面跟著小編一起來看下吧
    2017-03-03
  • Python 自動登錄淘寶并保存登錄信息的方法

    Python 自動登錄淘寶并保存登錄信息的方法

    這篇文章主要介紹了Python 自動登錄淘寶并保存登錄信息的方法,本文圖文并茂給大家介紹的非常詳細,具有一定的參考借鑒價值,需要的朋友可以參考下
    2019-09-09
  • python+opencv實現(xiàn)文字顏色識別與標定功能

    python+opencv實現(xiàn)文字顏色識別與標定功能

    最近小編接了一個比較簡單的圖像處理的單子,今天小編給大家分享python+opencv實現(xiàn)文字顏色識別與標定功能的完整思路及代碼,感興趣的朋友一起看看吧
    2021-09-09
  • python繪制折線圖和條形圖的方法

    python繪制折線圖和條形圖的方法

    這篇文章主要為大家詳細介紹了python實現(xiàn)折線圖和條形圖,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2022-04-04
  • 如何將numpy二維數(shù)組中的np.nan值替換為指定的值

    如何將numpy二維數(shù)組中的np.nan值替換為指定的值

    這篇文章主要介紹了將numpy二維數(shù)組中的np.nan值替換為指定的值操作,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2021-05-05
  • python卸載numpy出現(xiàn)WinError:拒絕訪問的解決方案

    python卸載numpy出現(xiàn)WinError:拒絕訪問的解決方案

    這篇文章主要介紹了python卸載numpy出現(xiàn)WinError:拒絕訪問的解決方案,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教
    2023-08-08
  • python字符串的常用操作方法小結(jié)

    python字符串的常用操作方法小結(jié)

    這篇文章主要為大家詳細介紹了python字符串的常用操作方法,如字符串的替換、刪除、截取、復(fù)制、連接、比較、查找、分割等,需要的朋友可以參考下
    2016-05-05
  • Python后臺管理員管理前臺會員信息的講解

    Python后臺管理員管理前臺會員信息的講解

    今天小編就為大家分享一篇關(guān)于Python后臺管理員管理前臺會員信息的講解,小編覺得內(nèi)容挺不錯的,現(xiàn)在分享給大家,具有很好的參考價值,需要的朋友一起跟隨小編來看看吧
    2019-01-01

最新評論