一篇文章带你了解Django ORM操作(进阶篇)

Karen110
• 阅读 1215

回顾

上次咱们学习了一下Django ORM的基本查询操作。

查询操作主要使用的是filter()方法。

我们知道filter()查询出来的是值,如果想取第一个值需要再filter().first()才行。

还知道了get()filter().first()的区别等等。

Django ORM的查询还有很多,继续来看叭!!!

查询操作

对象.外键字段

比如,我们拿到了一个书的信息,我们可以这样打印他的信息。

代码


book = models.Book.objects.filter(title="<<大明帝国>>").first()
print(f"book类型:{type(book)}")
print(f"id:{book.id}")
print(f"书名:{book.title}")
print(f"价格:{book.price}")
print(f"书名:{book.PublishDate}")
print(f"出版社:{book.publish}") # 外键字段

执行结果

一篇文章带你了解Django ORM操作(进阶篇)

注:蓝色为外键字段

不知道你有没有疑问,为什么book.publish会把邮电出版社打印出来

这个原因主要在于外键对象的__str__方法。

一篇文章带你了解Django ORM操作(进阶篇)

就是因为我Publish返回的是self.title,所以才能打印出来邮电出版社,如果我想打印出版社联系方式咋办?

代码

print(f"出版社类型:{type(book.publish)}")  # <class 'web.models.Publish'>
# book.publish已经是models.Publish对象,所以可以自由调里面的属性
print(f"出版社电话:{book.publish.phone},")

执行结果

一篇文章带你了解Django ORM操作(进阶篇)

总结

对象.外键字段拿到的就是外键字段对象,直接就可以通过对象.外键字段.外键属性获取具体值。

反向查询(表名__set.all())

上述我们是通过正向查询的方式查询到了书对应的出版社具体信息

但是如果说,我们拿到的就是一个出版社名呢?

通常情况下,你可能会这样!

代码

# 查询邮电出版社
publish_obj = models.Publish.objects.filter(title="邮电出版社").first()
# 获取出版社id
publish_id = publish_obj.id
# 查询publish_id为出版社id的
book_list = models.Book.objects.filter(publish_id=publish_id)
print(book_list)

执行结果

一篇文章带你了解Django ORM操作(进阶篇)

其实,还有一种方法:通过一个对象,反向查多个对象。

代码

publish_obj = models.Publish.objects.filter(title="邮电出版社").first()
book_list = publish_obj.book_set.all()
print(book_list)

执行结果

一篇文章带你了解Django ORM操作(进阶篇)

双下划线跨表查询

还是上述这个问题,通过一个出版社名,查找属于这个出版社的图书

基于双下划线的跨表查询,理论是更简单的!

一篇文章带你了解Django ORM操作(进阶篇)

注: 可以看到还有__contains等其他filter条件查询,通过__跨表依然是可以通用的。

代码


book_list = models.Book.objects.filter(publish__title="邮电出版社")
print(book_list)

执行结果

一篇文章带你了解Django ORM操作(进阶篇)

连续跨表

__不仅可以进行跨一张表,还能跨多张表。

图书Many作者表为例,根据出版社查询图书和作者多对多的信息。

代码


ret = models.BookManyAuthor.objects.filter(book__publish__title="邮电出版社")
print(ret)

跨了book表又跨了publish表

一篇文章带你了解Django ORM操作(进阶篇)

执行结果

一篇文章带你了解Django ORM操作(进阶篇)

values

有时候,我们可能只需要一些特定的列,这时候使用values即可。

代码


# 语法
book_list = models.Book.objects.all().values("列1","列2",...)
# 示例
book_list = models.Book.objects.all().values("title","price")
print(book_list)

代码

一篇文章带你了解Django ORM操作(进阶篇)

values返回的值有点像列表套字典,但是其实本质还是QuerySet类型。

values_list

values_listvalues功能一样,都是取相关的列,但是返回的类型格式不一样。

代码


book_list = models.Book.objects.all().values_list("title","price")
print(book_list)

执行结果

一篇文章带你了解Django ORM操作(进阶篇)

这个有点像列表套元组,但是其实本质还是QuerySet

related_name通常用于反向查询时,替换<表名>_set

原方式

models.py

一篇文章带你了解Django ORM操作(进阶篇)

代码


# 查询邮电出版社
publish = models.Publish.objects.filter(title="邮电出版社").first()
print(publish)
# 反向一对多
book_list = publish.book_set.all()
print(book_list)

执行结果

一篇文章带你了解Django ORM操作(进阶篇)

别名方式

models.py

一篇文章带你了解Django ORM操作(进阶篇)

代码


# 查询邮电出版社
publish = models.Publish.objects.filter(title="邮电出版社").first()
print(publish)
# 反向一对多
book_list = publish.book_list.all()
print(book_list)```

**执行结果**  

![](https://img-hello-world.oss-cn-beijing.aliyuncs.com/01c173c215ed5e11b897ddea43fa287e.png)

### filter().filter()...  

上文我们说过,是支持多个filter的,`filter(<条件>).filter(<条件>)...`

这种情况通常用于**不确定筛选条件**,但是**多层筛选**的情况下。

**代码**

举例而已,后面filter里面可以是其他 或 的条件

book1 = models.Book.objects.filter(title="<<大明帝国>>").filter(price="99")

效果同上

book2 = models.Book.objects.filter(title="<<大明帝国>>",price="99") print(book1) print(book2)

```

执行结果

一篇文章带你了解Django ORM操作(进阶篇)

总结


本篇主要还是上篇的继续补充,还是关于filter的查询部分。

本次主要有外键字段类型,反向查询默认使用<表名>__set,还可以使用related_name反向字段查询。

双下划线可以进行条件查询,还可以进行跨表查询,还可以连续跨表,values和values_list区别。

多个filter进行条件筛选。

如果在操作过程中有任何问题,记得下面留言,我们看到会第一时间解决问题。

用微笑告诉别人,今天的我比昨天强,今后也一样。

如果觉得还不错,记得动手点赞一下哈。感谢你的观看。

如果你觉得文章还可以,记得点赞留言支持我们哈。感谢你的阅读,有问题请记得在下方留言噢~

想学习更多关于Python的知识,可以参考学习网址:http://pdcfighting.com/,点击阅读原文,可以直达噢~

**-----**------**-----**---**** End **-----**--------**-----**-****

往期精彩文章推荐:

一篇文章带你了解Django ORM操作(进阶篇)

欢迎各位大佬点击链接加入群聊【helloworld开发者社区】:https://jq.qq.com/?_wv=1027&k=mBlk6nzX进群交流IT技术热点。

本文转自 https://mp.weixin.qq.com/s/dkofEfG81isLYt2l-Ln8eA,如有侵权,请联系删除。

点赞
收藏
评论区
推荐文章
blmius blmius
3年前
MySQL:[Err] 1292 - Incorrect datetime value: ‘0000-00-00 00:00:00‘ for column ‘CREATE_TIME‘ at row 1
文章目录问题用navicat导入数据时,报错:原因这是因为当前的MySQL不支持datetime为0的情况。解决修改sql\mode:sql\mode:SQLMode定义了MySQL应支持的SQL语法、数据校验等,这样可以更容易地在不同的环境中使用MySQL。全局s
皕杰报表之UUID
​在我们用皕杰报表工具设计填报报表时,如何在新增行里自动增加id呢?能新增整数排序id吗?目前可以在新增行里自动增加id,但只能用uuid函数增加UUID编码,不能新增整数排序id。uuid函数说明:获取一个UUID,可以在填报表中用来创建数据ID语法:uuid()或uuid(sep)参数说明:sep布尔值,生成的uuid中是否包含分隔符'',缺省为
Jacquelyn38 Jacquelyn38
3年前
2020年前端实用代码段,为你的工作保驾护航
有空的时候,自己总结了几个代码段,在开发中也经常使用,谢谢。1、使用解构获取json数据let jsonData  id: 1,status: "OK",data: 'a', 'b';let  id, status, data: number   jsonData;console.log(id, status, number )
Karen110 Karen110
3年前
​一篇文章总结一下Python库中关于时间的常见操作
前言本次来总结一下关于Python时间的相关操作,有一个有趣的问题。如果你的业务用不到时间相关的操作,你的业务基本上会一直用不到。但是如果你的业务一旦用到了时间操作,你就会发现,淦,到处都是时间操作。。。所以思来想去,还是总结一下吧,本次会采用类型注解方式。time包importtime时间戳从1970年1月1日00:00:00标准时区诞生到现在
Stella981 Stella981
3年前
Python3:sqlalchemy对mysql数据库操作,非sql语句
Python3:sqlalchemy对mysql数据库操作,非sql语句python3authorlizmdatetime2018020110:00:00coding:utf8'''
Wesley13 Wesley13
3年前
mysql设置时区
mysql设置时区mysql\_query("SETtime\_zone'8:00'")ordie('时区设置失败,请联系管理员!');中国在东8区所以加8方法二:selectcount(user\_id)asdevice,CONVERT\_TZ(FROM\_UNIXTIME(reg\_time),'08:00','0
Wesley13 Wesley13
3年前
00:Java简单了解
浅谈Java之概述Java是SUN(StanfordUniversityNetwork),斯坦福大学网络公司)1995年推出的一门高级编程语言。Java是一种面向Internet的编程语言。随着Java技术在web方面的不断成熟,已经成为Web应用程序的首选开发语言。Java是简单易学,完全面向对象,安全可靠,与平台无关的编程语言。
Stella981 Stella981
3年前
Django中Admin中的一些参数配置
设置在列表中显示的字段,id为django模型默认的主键list_display('id','name','sex','profession','email','qq','phone','status','create_time')设置在列表可编辑字段list_editable
Wesley13 Wesley13
3年前
MySQL部分从库上面因为大量的临时表tmp_table造成慢查询
背景描述Time:20190124T00:08:14.70572408:00User@Host:@Id:Schema:sentrymetaLast_errno:0Killed:0Query_time:0.315758Lock_
Python进阶者 Python进阶者
9个月前
Excel中这日期老是出来00:00:00,怎么用Pandas把这个去除
大家好,我是皮皮。一、前言前几天在Python白银交流群【上海新年人】问了一个Pandas数据筛选的问题。问题如下:这日期老是出来00:00:00,怎么把这个去除。二、实现过程后来【论草莓如何成为冻干莓】给了一个思路和代码如下:pd.toexcel之前把这