Django多层嵌套ManyToMany字段ORM操作详解


Posted in Python onMay 19, 2020

在用django写项目时,遇到了许多场景,关于ORM操作获取数据的,但是不好描述出来,百度搜索关键词都不知道该怎么搜,导致一个人鼓捣了好久。这里细化下问题,还原场景,记录踩下的坑

首先先列举model,我举些生活中的例子,更方便理解问题

# 习题
class Problem(models.Model):
  desc = models.CharField()
  answer = models.TextField()
  is_pass = models.BooleanField(default=False, verbose_name="是否通过")

# 章节
class Chapter(models.Model):
  _id = models.IntegerField(verbose_name="编号")
  title = models.CharField()
  problem = models.ManyToManyField(Problem)
  pass_rate = models.IntegerField(verbose_name="通关率")

# 书籍  
class Book(models.Model):
  title = models.CharField()
  desc = models.TextField()
  chapter = models.ManyToManyField(Chapter,verbose_name="章节")
  speed = models.IntegerField(verbose_name="学习进度", default=0)

假设是一本数学书,有5个章节,每个章节里有数量不等的习题,

即book与chapter是多对多,chapter与problem也是多对多

场景一: 书籍下的所有习题

# 按我的理解是取问题非空的章节数
# 类似于问爷爷有几个孙子,没办法跨辈,就按一个孙子对应一个爸爸来取(有重复)
book.chapter.filter(problem___id__isnull=False).count()

场景二:书籍下所有通过的习题

book.chapter.filter(problem__is_pass=True).count()

场景三: 判断某个问题是否在这本书里

def problem_in_ladder(book, problem):
    for i in book.chapter.all():
      if problem in i.problem.all():
        return True
    return False

尽可能的减少view中对models的取值操作,所以把上面几个场景方法写在models类中

最终的models

# 习题
class Problem(models.Model):
  desc = models.CharField()
  answer = models.TextField()
  is_pass = models.BooleanField(default=False, verbose_name="是否通过")

# 章节
class Chapter(models.Model):
  _id = models.IntegerField(verbose_name="编号")
  title = models.CharField()
  problem = models.ManyToManyField(Problem)
  pass_rate = models.IntegerField(verbose_name="通关率")
 
  @property
  def items(self):
    return self.problem.count()

  @property
  def pass_problem(self):
    return self.problem.filter(is_pass=True).count()
  
# 书籍  
class Book(models.Model):
  title = models.CharField()
  desc = models.TextField()
  chapter = models.ManyToManyField(Chapter,verbose_name="章节")
  speed = models.IntegerField(verbose_name="学习进度", default=0)
  
  @property
  def chapters(self):
    return self.chapter.count()

  @property
  def pass_count(self):
    return self.chapter.filter(problem__is_pass=True).count()

  @property
  def items(self):
    return self.chapter.filter(problem___id__isnull=False).count()

补充知识:django中当model设置了ordering后,使用distinct()和annotate()问题记录

model类如下,我在class Meta中设置了ordering = ['-date_create'],即模型对象返回的记录结果集是按照这个字段排序的。

class SystemUserPushHistory(models.Model):
 
  id = models.UUIDField(default=uuid.uuid4, primary_key=True)
  host_name = models.CharField(max_length=128, null=False)
  system_username = models.CharField(max_length=128, null=False)
  method = models.CharField(max_length=32, null=False)
  is_success = models.BooleanField(default=False)
  date_create = models.DateTimeField(auto_now_add=True, editable=False)
  message = models.CharField(max_length=4096, null=True)
 
  class Meta:
    db_table = "assets_systemuser_push_history"
    ordering = ['-date_create']
 
  def __str__(self):
    ret = self.system_username + " => " + self.host_name
    return ret

当业务有需求如对host_name进行分组显示,在代码中用到了annotate,如下。

>>> from django.db.models import Count 
>>> from assets.models import SystemUserPushHistory
>>> p = SystemUserPushHistory.objects.values("host_name").annotate(dcount=Count(1))
>>> p
<QuerySet [{'host_name': '点2', 'dcount': 1}, {'host_name': '点3', 'dcount': 2}, {'host_name': '点2', 'dcount': 1}, {'host_name': '点3', 'dcount': 1}]>
>>> print(p.query)
SELECT `assets_systemuser_push_history`.`host_name`, COUNT(1) AS `dcount` FROM `assets_systemuser_push_history` GROUP BY `assets_systemuser_push_history`.`host_name`, `assets_systemuser_push_history`.`date_create` ORDER BY `assets_systemuser_push_history`.`date_create` DESC

可以看到,所得到的结果并不像我们预期的一样,之后把执行的sql输出出来可以看到在group by的时候是对host_name和date_create进行分组,原因就是因为我们在model类中设置了ordering,去掉之后代码运行正常。

使用distinct和上面的情况类似,就不列出来了。

以上这篇Django多层嵌套ManyToMany字段ORM操作详解就是小编分享给大家的全部内容了,希望能给大家一个参考,也希望大家多多支持三水点靠木。

Python 相关文章推荐
使用Python实现BT种子和磁力链接的相互转换
Nov 09 Python
Python+Socket实现基于TCP协议的客户与服务端中文自动回复聊天功能示例
Aug 31 Python
python实现串口自动触发工作的示例
Jul 02 Python
django多个APP的urls设置方法(views重复问题解决)
Jul 19 Python
Windows10下 python3.7 安装 facenet的教程
Sep 10 Python
python+Django实现防止SQL注入的办法
Oct 31 Python
python 循环数据赋值实例
Dec 02 Python
python3 字符串知识点学习笔记
Feb 08 Python
Python爬虫之Spider类用法简单介绍
Aug 04 Python
python+pygame实现坦克大战小游戏的示例代码(可以自定义子弹速度)
Aug 11 Python
python简单实现9宫格图片实例
Sep 03 Python
python 实现全球IP归属地查询工具
Dec 18 Python
django ORM之values和annotate使用详解
May 19 #Python
基于python实现地址和经纬度转换
May 19 #Python
Python Django form 组件动态从数据库取choices数据实例
May 19 #Python
Django自关联实现多级联动查询实例
May 19 #Python
Python的Django框架实现数据库查询(不返回QuerySet的方法)
May 19 #Python
django 数据库返回queryset实现封装为字典
May 19 #Python
使用PyQt的QLabel组件实现选定目标框功能的方法示例
May 19 #Python
You might like
通过ICQ网关发送手机短信的PHP源程序
2006/10/09 PHP
ECMall支持SSL连接邮件服务器的配置方法详解
2014/05/19 PHP
PHP根据两点间的经纬度计算距离
2014/10/31 PHP
php传值赋值和传地址赋值用法实例分析
2015/06/20 PHP
php自动加载方式集合
2016/04/04 PHP
基于PHP常用文件函数和目录函数整理
2017/08/17 PHP
理解Javascript_08_函数对象
2010/10/15 Javascript
js操作iframe的一些方法介绍
2013/06/25 Javascript
js 控制页面跳转的5种方法
2013/09/09 Javascript
JavaScript实现三阶幻方算法谜题解答
2014/12/29 Javascript
Javascript通过overflow控制列表闭合与展开的方法
2015/05/15 Javascript
javascript显示中文日期的方法
2015/06/18 Javascript
Javascript实现字数统计
2015/07/03 Javascript
使用Node.js给图片加水印的方法
2016/11/15 Javascript
JS实现十分钟倒计时代码实例
2018/10/18 Javascript
vuex 中插件的编写案例解析
2019/06/10 Javascript
angular6开发steps步骤条组件
2019/07/04 Javascript
JS组件库AlloyTouch实现图片轮播过程解析
2020/05/29 Javascript
JS实现多选框的操作
2020/06/24 Javascript
JavaScript中交换值的10种方法总结
2020/08/18 Javascript
从源码角度来回答keep-alive组件的缓存原理
2021/01/18 Javascript
web.py在模板中输出美元符号的方法
2014/08/26 Python
Python实现JSON反序列化类对象的示例
2018/01/31 Python
Python实现加载及解析properties配置文件的方法
2018/03/29 Python
python logging设置level失败的解决方法
2020/02/19 Python
Python描述符descriptor使用原理解析
2020/03/21 Python
Pandas将列表(List)转换为数据框(Dataframe)
2020/04/24 Python
python pandas dataframe 去重函数的具体使用
2020/07/20 Python
Pycharm2020最新激活码|永久激活(附最新激活码和插件的详细教程)
2020/09/29 Python
Pytorch1.5.1版本安装的方法步骤
2020/12/31 Python
施华洛世奇澳大利亚官网:SWAROVSKI澳大利亚
2017/01/06 全球购物
办加油卡单位介绍信
2014/01/09 职场文书
个人租房协议书
2014/11/28 职场文书
团代会邀请函
2015/02/02 职场文书
2015年乡镇纪检工作总结
2015/04/22 职场文书
同意离婚答辩状
2015/05/22 职场文书