Django 联表查询操作

news2024/12/24 9:18:44

在日常的开发中,常常需要对多张数据表同时进行数据查询。多表查询需要在数据表之间建立表关系才能够实现。一对多或一对一的表关系是通过外键实现关联的,而多表查询分为正向查询和反向查询。

表模型结构

以歌手表、专辑表、单曲表查询为例子。

歌手与专辑为一对多关系;歌手和单曲为一对多关系;专辑和单曲为多对多关系;

表模型如下:

class Singler(BaseModel):
    """ 歌手表模型 """

    name = models.CharField(max_length=50)
    first_letter = models.CharField(max_length=15, editable=False)
    # 设置上传位置
    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)
    # 设置与歌手表关联外键 一对多外键设置在多的模型中
    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,)

    # 设置与歌手表关联外键 一对多
    singler = models.ForeignKey("Singler",on_delete=models.CASCADE)

    # 设置与单曲表关联外键 多对多
    Singe = models.ManyToManyField('Singe')

 

单曲获取歌手

通过singe模型关联外键singler获取关联歌手Singler信息,为正向查询。

代码如下:

info = Singe.objects.filter(id=1).first()
print('单曲信息:', info)
article = info.singler
print('歌手信息:', article)

效果:

歌手获取单曲

通过歌手模型获取单曲相应记录,因为外键在单曲表模型中,

这样属于反向查询。

方法一

使用小写模型名_set方式查询。

代码如下

info = Singler.objects.filter(id=1).first()
print('歌手信息:', info)
song = info.singe_set.first()
print('一首单曲:', song)
songs = info.singe_set.all()
print('全部单曲:', songs)

方法二

需要对外键设置related_name为某个字符串,来进行关联查询。

模型外键设置
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)

效果:

查询关联条件记录

正向查询

通过单曲表查询歌手名称是周杰伦的单曲和歌手信息。

代码如下:

info = Singe.objects.filter(singler__name='周杰伦').first()
print('单曲信息:', info)
article = info.singler
print('歌手信息:', article)

singler是关联外键,name是歌手表name字段,两者使用双下划连接;

singler是Singler模型在Singe模型中设置的外键。

效果:

反向查询

通过歌手表查询歌曲名称获取歌手信息和单曲信息。

代码如下:

info = Singler.objects.filter(singler_info__name='告白气球').first()
print('歌手信息:', info)
song = info.singler_info.first()
print('单曲信息:', song)

singler_info是models.py中表模型外键设置的属性related_name='singler_info'。

通过单曲名称获取相应单曲的歌手信息,

之后通过参数singler_info反向获取模型Singe的数据。

效果:

联表查询优化

无论是正向查询还是反向查询,它们在数据库里需要执行两次SQL查询,第一次是查询某张数据表的数据,再通过外键关联获取另一张数据表的数据信息。为了减少查询次数,提高查询效率,我们可以使用select_related或prefetch_related方法实现,该方法只需执行一次SQL查询就能实现多表查询。

Select_related

select_related主要针对一对一和一对多关系进行优化,它是使用SQL的JOIN语句进行优化的,通过减少SQL查询的次数来进行优化和提高性能。

正向查询

select_related方法,参数为字符串格式,以模型Singe为查询对象;

select_related使用INNER JOIN方式查询两个数据表;

查询模型Singe的字段singler和模型Singler的字段id;

select_related参数为singler为外键字段;

若要得到其他数据表的关联数据,则可用双下画线“__”连接字段名;

双下画线“__”连接字段名必须是外键字段名或外键字段related_name设置参数。

代码如下:

p = Singe.objects.select_related('getname').
values('id', 'name', 'duration', 'singler__name')
# # 查看SQL查询语句
print(p.query)
# 查看结果 为dict格式
print(p)

效果:

反向查询

以模型Vocation为查询对象

select_related使用LEFT JOIN方式查询两个数据表

select_related的参数为related_name设置参数,属于关联表字段。

代码如下:

f = Singler.objects.select_related('getname').
values('id', 'name', 'getname__name')
# 查看SQL查询语句
print(f.query)
# 查看结果
print(f)

print('#'*100)

# 获取两个模型的数据,以模型Singler的singe_num大于1为查询条件
f = Singler.objects.select_related('getname').
filter(singe_num__gt=1).values('id', 'name', 'getname__name')
# 查看SQL查询语句
print(f.query)
# 获取查询结果集的首个元素的字段getname__name的值
print(f[0]['getname__name'])

效果:

Prefetch_related

prefetch_related和select_related的设计目的很相似,都是为了减少SQL查询的次数,但是实现的方式不一样。select_related是由SQL的JOIN语句实现的,但是对于多对多关系,使用select_related会增加数据查询时间和内存占用;而prefetch_related是分别查询每张数据表,然后由Python语法来处理它们之间的关系,因此对于多对多关系的查询,prefetch_related更有优势。

设置related_name
# 设置与单曲表关联外键 多对多
Singe = models.ManyToManyField(
    'Singe',
    verbose_name='单曲',
    help_text='请选择单曲',
    related_name='singe_info'
)

 

视图处理

Album模型与Singe模型关系为多对多,也就是专辑可以添加多个单曲,单曲也可以加入多个专辑。查询单曲名称为告白气球的加入了哪些专辑。

代码如下:

s = Singe.objects.prefetch_related('singe_info').filter(name='告白气球').first()
print(s)
# # 根据外键字段singe获取当前数据的多对多或一对多关系
print(s.singe_info.all())

print('#'*100)


# 使用values_list获取联合查询数据
s = Singe.objects.prefetch_related('singe_info').filter(name='告白气球')\
    .values_list('id', 'name', 'singe_info__name')
# 查看sql
print(s.query)
# 查看结果
print(s)
# 输出专辑名
print(s[0][2])

效果:

执行sql语句

也可以通过raw方式,将查询条件使用原生SQL语法实现,

此方法需要依靠模型对象,在某程度上可防止SQL注入。

Raw查询所有

单曲表和歌手表联查,查询所有数据。

代码如下:

s1 = Singe.objects.raw('select * from player_singe as s 
left join player_singler as a on s.singler_id = a.id')
print('查询结果')
print(s1)
for item in s1:
    print(item)

效果:

Raw条件查询

单曲表和歌手表联查,查询单曲名称为‘告白气球’。

代码如下:

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('查询结果')
print(s.query)
print(s)
print(s[0])

效果:

 

Execute查询

execute执行SQL语句无须经过Django的ORM框架。借助第三方模块实现连接过程,如MySQL的mysqlclient模块和SQLite的sqlite3模块等,这些模块连接数据库之后,可通过游标的方式来执行SQL语句。很容易受到SQL注入攻击,需要自己做参数的验证和过滤操作。

代码如下:

from django.db import connection
cursor = connection.cursor()
# 执行SQL语句
cursor.execute('select * from player_singe as s left join
 player_singler as a on s.singler_id = a.id')
# 读取第一行数据
print(cursor.fetchone())
# 读取所有数据
print(cursor.fetchall())

效果:

 

总结

作为一个django使用的新手,在做练手项目中对联表查询感觉比较生疏,最近两天整理了一些连表查询应用场景和使用方法;以及无法使用django中ORM操作的原生查询,以备之后忘记用作参考使用。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1045195.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

RK3588 VDD_LOGIC电源PCB设计注意事项

RK3588 VDD_LOGIC电源PCB设计 1、VDD_LOGIC的覆铜宽度需满足芯片的电流需求,连接到芯片电源管脚的覆铜足够宽,路径不能被过孔分割太严重,必须计算有效线宽,确认连接到CPU每个电源PIN脚路径都足够。 2、如图1所示,原理…

Scrapy-应对反爬虫机制

参考自https://blog.csdn.net/y472360651/article/details/130002898 记得把BanSpider改成自己的项目名,还有一个细节要改一下,把代码user换成user_agent 禁止Cookie 在Scrapy项目中的settings文件,可以发现文件中有以下代码: COOKIES_ENA…

红黑树-自平衡二叉搜索树

一、简介 红黑树(Red-Black Tree)是一种自平衡的二叉搜索树,它的节点可以是红色或黑色。这个颜色的设计是为了满足红黑树的五个关键性质,确保树保持平衡和高效地支持插入、删除和搜索操作。 以下是红黑树的五个关键性质&#xf…

【Unity3D日常开发】Unity3D中Quality的设置参考

推荐阅读 CSDN主页GitHub开源地址Unity3D插件分享简书地址我的个人博客 大家好,我是佛系工程师☆恬静的小魔龙☆,不定时更新Unity开发技巧,觉得有用记得一键三连哦。 一、前言 这篇文章就来讲一下Quality的设置(Unity版本&#…

浪潮信息被Gartner评为全球文件存储标杆厂商

近日,国际权威研究机构 Gartner 正式发布《2023年存储和数据保护技术成熟度曲线报告》((Hype Cycle for Storage and Data Protection Technologies, 2023,以下简称“报告”),基于对市场应用的前瞻洞察和多年的技术精耕…

518抽奖软件,一键打印中奖名单的方法和用途

518抽奖软件简介 518抽奖软件,518我要发,超好用的年会抽奖软件,简约设计风格。 包含文字号码抽奖、照片抽奖两种模式,支持姓名抽奖、号码抽奖、数字抽奖、照片抽奖。(www.518cj.net) 一键打印中奖名单 主窗口上按 CtrlP 打开 {…

在关系型数据库中储存树形结构

adjacency list 邻接表显然是最简单的方式,也是在实践中经常用到的。其储存节点以及直接父节点来进行储存树形结构 邻接表结构简单,查询修改节点的直接父节点都很容易。然而如果返回父节点下的所有节点之类的跨层操作那就很麻烦了,需要频繁…

matlab实现杨氏双缝干涉实验可视化界面

关于杨氏双缝干涉实验的条纹光强理论推导和matlab绘图可以参考下面的链接:杨氏双缝干涉实验matlab实现 接下来利用GUI实现可视化界面。 一、GUI GUIDE简介 1、在命令行窗口输入小写的guide可以自动弹出fig窗口。 2、界面的左侧是常用的工具,鼠标悬停…

利用串口示波器调试PID参数

PID调试最麻烦的是参数调整,需要花费大量时间,如果每调整一次都要修改代码重新编译烧录,效率很低,推荐串口示波器与rt-thead finsh进行调试 推荐使用FireWater数据引擎 定时上报数据,rt_kprintf不支持浮点数据打印&a…

从零开始之了解电机及其控制(6)六步换向法

引导:六步换向的本质是? 因为无刷电机有三根线,而H桥可以将负载连接到正电压或者地,于是用三对MOS管组成的H桥驱动电机,称为半桥驱动。 无刷电机的优点如下: 首先,由于所有三个电机相位都通过…

xcode15下载ios17模拟器失败

升级到xcode15后需要安装ios17模拟器 但是在下载过程中会遇到报错 如下图这种 网上搜索了一下发现有人遇到过无法下载的问题,并且在apple官网也有人提出类似问题 https://developer.apple.com/forums/thread/737648 解决方案就是从https://developer.apple.com/do…

linux提权秘籍

Linux 提权总结 一、常用基础 1、自定义可执行文件(Custom Executable) 可能有某些根进程执行另一个可以控制的进程。在这些情况下,以下C代码一旦编译,将生成一个作为根运行的sbashell: int main() { setuid(0);system("/bin/bash -p…

MapBox GL JS出现“Unimplemented type: 7”问题的解决办法

Mapbox GL JS在矢量瓦片的渲染方面有独特的优势,可以支持动态的样式,支持字体切片,快速加载各种字体。使用起来十分方便,但是在很长的一段时间内,经常遇到出现大量“Unimplemented type: 7”的控制台错误提示&#xff…

Web自动化测试 —— headless无头浏览器!

一、Options概述 是一个配置浏览器启动的选项类,用于自定义和配置Driver会话常见使用场景: 设置无头模式:不会显示调用浏览器,避免人为干扰的问题。设置调试模式:调试自动化测试代码(浏览器复用) 二、添加启动配置 添…

丰田 Auris 混动车电池冷却系统异常

故障现象 一辆丰田 Auris 混合动力车行驶时,混合动力车警告灯亮起。这辆车被改装成监控用车,车厢内到处都装有敏感的录音设备。 为了不被下面的情况所影响,我们从混合动力控制单元(HCU)中提取了故障代码 P0A82-123。混…

Multisim14.0仿真(二十四)基于LM555定时器的施密特触发器

一、仿真原理图: 二、仿真效果图:

【数据结构——单链表】本篇文章通过图文结合的方式能让你轻松的掌握单链表

链表的概念及结构 有了顺序表为什么还会出现链表呢? 链表和数组类似,但是功能比数组强大的多,数组的空间是固定的,在定义数组的时候空间大小就已经固定了,在使用时有可能会造成空间的浪费或者面临空间不够的风险&…

2023年建筑架子工(建筑特殊工种)证考试题库及建筑架子工(建筑特殊工种)试题解析

题库来源:安全生产模拟考试一点通公众号小程序 2023年建筑架子工(建筑特殊工种)证考试题库及建筑架子工(建筑特殊工种)试题解析是安全生产模拟考试一点通结合(安监局)特种作业人员操作证考试大纲和(质检局)特种设备作…

第九章 内存分区模型

C程序在执行时,将内存大方向划分为4个区域 代码区:存放函数体的二进制代码,由操作系统进行管理的 全局区:存放全局变量和静态变量以及常量 栈区:由编译器自动分配释放, 存放函数的参数值,局部变量等 堆区&#xff1…

【NLP的Python库(04/4)】:Flair

一、说明 Flair是一个现代的NLP库。从文本处理到文档语义,支持所有核心 NLP 任务。Flair使用现代转换器神经网络模型来完成多项任务,并结合了其他Python库,可以选择特定的模型。其清晰的API和注释文本的数据结构,以及多语言支持&a…