反向查询详解以Django为例
以下给出两张表格
class User(AbstractUser):mobile=models.CharField(max_length=11,default=0,unique=True,verbose_name='手机号')email_active=models.BooleanField(default=False,verbose_name='邮箱验证状态')default_address = models.ForeignKey('Address', related_name='users', null=True, blank=True,on_delete=models.SET_NULL, verbose_name='默认地址')class Meta:db_table = 'tb_users'verbose_name="用户"verbose_name_plural=verbose_namedef __str__(self):return self.usernamefrom utils.models import BaseModel
class Address(BaseModel):
"""用户地址"""user = models.ForeignKey(User, on_delete=models.CASCADE, related_name='addresses', verbose_name='用户')title = models.CharField(max_length=20, verbose_name='地址名称')receiver = models.CharField(max_length=20, verbose_name='收货人')province=models.ForeignKey('areas.Area', on_delete=models.PROTECT, related_name='province_addresses', verbose_name='省')city = models.ForeignKey('areas.Area', on_delete=models.PROTECT, related_name='city_addresses', verbose_name='市')district = models.ForeignKey('areas.Area', on_delete=models.PROTECT, related_name='district_addresses', verbose_name='区')place = models.CharField(max_length=50, verbose_name='地址')mobile = models.CharField(max_length=11, verbose_name='手机')tel = models.CharField(max_length=20, null=True, blank=True, default='', verbose_name='固定电话')email = models.CharField(max_length=30, null=True, blank=True, default='', verbose_name='电子邮箱')is_deleted = models.BooleanField(default=False, verbose_name='逻辑删除')class Meta:db_table = 'tb_address'verbose_name = '用户地址'verbose_name_plural = verbose_nameordering = ['-update_time']
问题集:
主键和外键名字为什么不是一模一样
主键是用于唯一标识表中一条记录的字段,而外键是用来建立表与表之间关联的字段。外键的名称通常不会和主键完全相同,主要是为了避免混淆,并且在一个表中可能会有多个外键指向同一张表。例如,在 Address 表中:
province、city、district 都是外键,指向 areas.Area 表。
它们分别表示 省、市、区,虽然都指向同一张表 areas.Area,但代表不同的地址层级,所以它们的名字必须不同,否则会引起歧义。
为什么province,city,district 是外键?
在 Address 表中,province、city 和 district 这三个字段都是指向 areas.Area 表的外键,它们的作用是让每个 Address 记录的省、市、区字段都与 Area 表中的相应区域关联起来。
假设 areas.Area 表的结构如下:
class Area(BaseModel): name = models.CharField(max_length=20, verbose_name="名称") parent = models.ForeignKey('self', null=True, blank=True, on_delete=models.SET_NULL, verbose_name="父级地区")
这样:
province 关联的是 Area 表中的 省级 数据;
city 关联的是 Area 表中的 市级 数据;
district 关联的是 Area 表中的 区县级 数据。
related_name 的作用
related_name 主要用于 Django 进行 反向查询,即:通过 外键关系 反向访问关联的表。默认情况下,Django 通过 外键名_set 进行反向查询,例如 area.address_set.all(),但 related_name 可以自定义这个反向查询的名称,提高代码的可读性和直观性。
在 Address 表中:
province = models.ForeignKey('areas.Area', on_delete=models.PROTECT, related_name='province_addresses', verbose_name='省') city = models.ForeignKey('areas.Area', on_delete=models.PROTECT, related_name='city_addresses', verbose_name='市') district = models.ForeignKey('areas.Area', on_delete=models.PROTECT, related_name='district_addresses', verbose_name='区')
表示:
province_addresses:查询所有 该省 下的地址
city_addresses:查询所有 该市 下的地址
district_addresses:查询所有 该区 下的地址
举例
假设 areas.Area 表中有以下数据:
| id | name | parent_id |
|---|---|---|
| 1 | 江苏省 | NULL |
| 2 | 南京市 | 1 |
| 3 | 玄武区 | 2 |
在 Address 表中,有如下数据:
| id | user_id | province | city | district | place |
|---|---|---|---|---|---|
| 1 | 10 | 1 | 2 | 3 | XX路100号 |
如何使用 related_name 进行查询?
# 获取江苏省(id=1)下的所有地址 province = Area.objects.get(id=1) print(province.province_addresses.all()) # 输出:<QuerySet [<Address: XX路100号>]> # 获取南京市(id=2)下的所有地址 city = Area.objects.get(id=2) print(city.city_addresses.all()) # 输出:<QuerySet [<Address: XX路100号>]> # 获取玄武区(id=3)下的所有地址 district = Area.objects.get(id=3) print(district.district_addresses.all()) # 输出:<QuerySet [<Address: XX路100号>]>
如果不使用 related_name,查询方法会变成:
print(province.address_set.all()) # 默认反向查询,名字不直观
所以 related_name 的作用是让查询逻辑更加清晰和直观。
2.什么是正向查询和反向查询
1. 正向查询
正向查询是指通过 外键字段从子表(即引用外键的表)查询父表(即外键所指向的表)中的数据。
在 Django 中,正向查询是从 子模型 到 父模型 的查询。例如,从 Address 表查询某个地址的 province(省),即通过 Address 中的 province 外键字段,查找 Area 表中的 name 字段。
正向查询的例子:
假设我们有以下模型:
class Area(models.Model): name = models.CharField(max_length=100) class Address(models.Model): user = models.ForeignKey(User, on_delete=models.CASCADE) province = models.ForeignKey(Area, on_delete=models.PROTECT) city = models.ForeignKey(Area, on_delete=models.PROTECT)
如果你想从 Address 查询 province 对应的 Area,你可以执行正向查询:
address = Address.objects.get(id=1) print(address.province.name) # 获取地址的省名
结果:
print(address.province.name) # 输出: 河北省
*2. 反向查询*
反向查询是指从 父表 查询到 子表 关联的数据。在 Django 中,当你在 父模型(即引用外键的表)上使用反向查询时,它会自动创建一个从父模型到子模型的查询方法。
例如,你可以通过 Area 查询所有与某个 province(省)相关的 Address 记录。这里,Area 是父表,而 Address 是子表。
在 Address 模型中,你定义了:
province = models.ForeignKey('Area', on_delete=models.PROTECT, related_name='province_addresses')
related_name='province_addresses' 是反向查询的名字,它允许你通过 Area 查询到所有与该 Area 关联的 Address 记录。
反向查询的例子:
从 Area 查询与之关联的所有 Address,例如查找所有与某个省(province_id)相关的地址:
province = Area.objects.get(id=130000) addresses_in_province = province.province_addresses.all()
结果:
for address in addresses_in_province: print(address.title, address.receiver)
相关文章:
反向查询详解以Django为例
以下给出两张表格 class User(AbstractUser):mobilemodels.CharField(max_length11,default0,uniqueTrue,verbose_name手机号)email_activemodels.BooleanField(defaultFalse,verbose_name邮箱验证状态)default_address models.ForeignKey(Address, related_nameusers, nullT…...
我们如何思考AI创业投资
🎬 Verdure陌矣:个人主页 🎉 个人专栏: 《C/C》 | 《转载or娱乐》 🌾 种完麦子往南走, 感谢您的点赞、关注、评论、收藏、是对我最大的认可和支持!❤️ 声明:本文作者转载,原文出自…...
详解在 MySQL 中建索引时的注意事项
MySQL 中建索引时的注意事项 1. 索引的必要性与设计2. 复合索引与列顺序3. 索引数量与维护4. 索引类型选择5. 特殊注意事项 1. 索引的必要性与设计 使用场景:优先为在 WHERE、JOIN、ORDER BY 和 GROUP BY 中频繁使用的列创建索引。合理的索引设计能显著提升查询效率…...
LabVIEW 中数字转字符串常用汇总
在 LabVIEW 编程环境里,数字与字符串之间的转换是一项极为基础且重要的操作,广泛应用于数据处理、显示、存储以及设备通信等多个方面。熟练掌握数字转字符串的方法和技巧,对编写高效、稳定的程序起着关键作用。接下来,我们将全面深…...
蓝桥杯 C/C++ 组历届真题合集速刷(二)
一、0ASC - 蓝桥云课 (单位换算)算法代码: #include <iostream> using namespace std; int main() {printf("%d",L);return 0; } 二、0时间显示 - 蓝桥云课 (单位换算)算法代码: #inclu…...
【接口自动化_数据格式与类型】
在HTTP接口的自动化测试中,请求的数据格式和内容类型是两个密切相关但又有所区别的概念。以下是它们的分类和详细说明: 一、数据格式 数据格式是指请求体(Body)中数据的组织方式,常见的数据格式有以下几种࿱…...
JavaScript/React中,...(三个连续的点)被称为 扩展运算符(Spread Operator) 或 剩余运算符(Rest Operator)
const processOrder (order) > {const tax order.total * 0.1;const finalAmount order.total tax;return { ...order, tax, finalAmount }; }; 解释一下,特别:...?在JavaScript/React中,...(三个连续的点)被称…...
网络带宽测速工具选择指南iperf3 nttcp tcpburn jperf使用详解
简介 本文主要介绍内网(局域网)与外网(互联网)的网络带宽测速工具下载地址、选择指南、参数对比、基本使用。 测速工具快速选择指南 测速工具下载地址 iperf 官网下载链接:iperf.fr/iperf-download.php该链接提供了不…...
源代码保密解决方案
背景分析 随着各行各业业务数据信息化发展,各类产品研发及设计等行业,都有关乎自身发展的核心数据,包括业务数据、源代码保密数据、机密文档、用户数据等敏感信息,这些信息数据有以下共性: — 属于核心机密资料&…...
网络安全小知识课堂(十二)
SQL 注入:一行代码如何毁掉整个数据库? 引言 想象一下:用户在一个搜索框中输入关键词,网站却突然崩溃,所有数据被清空 —— 这不是电影情节,而是 **SQL 注入攻击(SQL Injection)**…...
PyCharm使用Flask启动项目后,如何修改文件,开启启动加载或是热启动,不用重启项目,直接生效。
PyCharm使用Flask启动项目后,每次修改完文件比如html、py文件都要重启项目才生效,在测试时很不方便,如何设置热启动,修改完文件后直接生效了? 解决方法 1、app.py文件,设置debugTrue。开启调试模式。 开…...
SpringCloud微服务(一)Eureka+Nacos
一、认识 微服务技术对比: SpringCloud: 版本匹配: 二、服务拆分以及远程调用 消费者与提供者: Eureka: 搭建EurekaServer: Ribbon负载均衡: 实现原理: IRule:规则接口…...
【Java设计模式】第4章 简单工厂讲解
4. 简单工厂模式 4.1 简单工厂讲解 定义:由一个工厂对象决定创建哪种产品类的实例,属于创建型模式,但不属于GoF 23种设计模式。适用场景: 工厂类负责创建的对象较少。客户端仅需传入参数,无需关心对象创建逻辑。优点: 客户端只需传入参数即可获取对象,无需知道创建细节…...
深度学习天崩开局
李沐大神的d2l包导入, 这玩意需要python311版本,我现在版本已经313了,作为一个天生要强的男人,我是坚决不向低版本低头的。 然后我就研究啊,各种翻资料啊,然后deepseek加豆包都翻烂了, 最终所…...
【详细图文】在VScode中配置python开发环境
目录 一、下载安装VSCode 1、官网下载VSCode 2、安装VSCode 3、汉化vscode (1)已自动下载汉化版插件 (2)未自动下载汉化版插件 二、 下载安装Python 1、官网下载Python 2、安装Python (1)双击打开…...
Unity遮挡剔除与视锥剔除混合策略技术详解
一、核心剔除技术原理对比 1. 视锥剔除(Frustum Culling) 原理:根据物体包围盒与摄像机视锥体的相交测试,移除非可见物体 优势: 计算成本低(平均0.1ms/万物体) 完全自动执行 局限:…...
Java面试黄金宝典34
1. 主键索引底层的实现原理 定义 主键索引是数据库中用于唯一标识表中每一行记录的索引,常见的底层实现是 B 树结构。B 树是一种平衡的多路搜索树,由内部节点和叶子节点组成。内部节点只存储索引键和指向下一层节点的指针,不存储实际数据&am…...
VR体验馆如何用小程序高效引流?3步打造线上预约+团购裂变系统
VR体验馆如何用小程序高效引流?3步打造线上预约团购裂变系统 一、线上预约的核心价值:优化体验,提升转化 减少客户等待时间 通过小程序预约功能,客户可提前选择体验时段,避免到店排队。数据显示&#…...
使用 node.js 和 MongoDB 编写一个简单的增删改接口 demo
文章目录 前言一、环境准备二、项目结构三、环境变量四、连接数据库3.1. connect.js 文件 五、定义数据模型5.1. BannerModel.js 文件 六、实现 server 接口6.1. server.js 文件 七、服务文件7.1. app.js 文件 八、感谢 前言 Mongoose 是一个在 Node.js 环境中操作 MongoDB 数据…...
【基于Vue3组合式API的互斥输入模式实现与实践分享】
基于Vue3组合式API的互斥输入模式实现与实践分享 目录 背景与痛点设计思路技术实现使用场景与案例遇到的问题与解决方案最佳实践总结 1. 背景与痛点 在表单交互设计中,我们经常面临这样的场景:多种输入方式互斥。例如,在评分系统中&#…...
PCL拟合空间3D圆周 fit3DCircle
PCL版本 1.15.0 main.cpp #include<vector> #include<iostream> #include <array> #include <string> #include <windows.h> #include <omp.h> #include <charconv> // C17 #include <cstdlib> #include<chrono> #in…...
【Redis】——最佳实践
目录 一.键值设计 1.如何优雅的设计key结构 2.拒绝BigKey 3.选择合适的数据结构 4.总结 二.批处理优化(海量数据批处理) 1.Pipeline 2.集群模式下的批处理 三.服务端优化 1.持久化配置 2.慢查询问题 1.记录慢查询 2.找到慢查询 3.集群最佳…...
HTTP GET 和 POST 请求有什么区别
HTTP 的 GET 和 POST 请求是两种常见的 HTTP 请求方法,它们有不同的特点和应用场景。以下是它们的主要区别: 1. 用途 GET:用于从服务器获取数据或资源。GET 请求会附带查询参数在 URL 中,通常用于请求数据,如加载网页…...
Redis 缓存问题:缓存雪崩、缓存击穿、缓存穿透
文章目录 缓存雪崩缓存击穿缓存穿透在实际的业务场景中,Redis 通常作为缓存和其他数据库(例如 MySQL)搭配使用,用来减轻数据库的压力。但是在使用 Redis 作为缓存数据库的过程中,可能会遇到一些常见问题,例如缓存穿透、缓存击穿和缓存雪崩等。 缓存雪崩 缓存雪崩是指缓存…...
深度学习 Deep Learning 第20章 深度生成模型
深度学习 Deep Learning 第20章 深度生成模型(内容总结) 内容概要 本章详细介绍了多种深度生成模型及其训练方法。这些模型包括玻尔兹曼机(Boltzmann Machines)、受限玻尔兹曼机(RBM)、深度信念网络&…...
我提了一个 Androidx IssueTracker
问题 在运行 gradle plugin 插件的 transform R8 阶段出现了报错 Caused by: com.android.tools.r8.internal.xk: java.lang.NullPointerException: Cannot invoke “String.length()” because “” is null 报错日志 FAILURE: Build failed with an exception.* What went w…...
搭建复现环境
初始准备:安装配置搬运工 1,安装配置搬运工 这个流行的容器化工具。步骤如下: 更新软件源 apt-get update 编辑 安装搬运工 apt-get install 搬运工.io 编辑 2,修改搬运工的配置文件,添加内容 sudo systemctl d…...
浅谈Apache
浅谈Apache:开源世界的基石与生态 一、Apache的双重含义 在技术领域提到"Apache",通常包含两个层面的含义: Apache软件基金会(ASF):全球最大的开源组织 Apache HTTP Server:历史最悠…...
Docker全方位指南
目录 前言 第一部分:Docker基础与安装 1.1 什么是Docker? 1.2 Docker的适用场景 1.3 全平台安装指南 1.4 配置优化 第二部分:Docker核心操作与原理 2.1 镜像管理 2.2 容器生命周期 2.3 网络模型 2.4 Docker Compose 第三部分&…...
【SpringCloud】Nacos健康检查
5.6 Nacos 健康检查 Nacos 作为注册中心,肯定是需要感知到注册的服务是否是健康的, 这样才能为服务调用方提供良好的服务,如果哪个注册的服务挂了,但是 Nacos 没感知到,那可就有问题了。 5.6.1 健康检查机制 Nacos …...
