Django多层嵌套ManyToMany字段ORM操作详解
作者:宅神kin 发布时间:2023-07-01 02:26:28
在用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和上面的情况类似,就不列出来了。
来源:https://blog.csdn.net/weixin_42042680/article/details/84845741
猜你喜欢
- 引子例如,一个人可能会在计算机上存储大量的照片、视频和文档文件,这些文件可能散落在不同的文件夹中,难以管理和查找。该程序可以根据文件类型将这
- 项目现状项目是一个数据监测平台,引入了ehcart和three.js 负责项目的数据可视化;打包后,体积高达2.1M,这个体积相比于我的项目
- Golang 支持交叉编译,在一个平台上生成另一个平台的可执行程序,最近使用了一下,非常好用,这里备忘一下。Mac 下编译 Linux 和
- 我的测试环境是2000sever ie6.0+sp4 MYIE1.31 (成功通过测试)关闭窗口的途径常用4种:1.双击左上角图标2.直接双
- 众所周知,由于 GIL 的存在,Python 单进程中的所有操作都是在一个CPU核上进行的,所以为了提高运行速度,我们一般会采用多进程的方式
- 1. 概述information_schema 数据库跟 performance_schema 一样,都是 MySQL 自带的信息数据库。其
- 一、了解seleniumSelenium是一个用于测试网站的自动化测试工具,支持各种浏览器包括Chrome、Firefox、Safar等浏览
- keras中卷积层Conv2D的学习关于卷积的具体操作不细讲,本文只是自己太懒了不想记手写笔记。由于自己接触到的都是图像处理相关的工作,因此
- 花了两周时间,利用工作间隙时间,开发了一个基于Django的项目任务管理Web应用。项目计划的实时动态,可以方便地被项目成员查看(^_^又重
- 今天写项目的时候用到ant design中的日期组件,但是由于用ant design日期组件取得的值是moment类型,而往数据库中保存需要
- 具体代码如下所示:package mainimport ( "bufio" "io&q
- 本文实例分析了javascript定义变量时带var与不带var的区别。分享给大家供大家参考。具体分析如下:直接看实例里说明:<scr
- 前言平时我们在接收后端返回的json对象通常是一个字符串类型的object,所以一般我们要对这个object进行类型转化后,我们才能使用ob
- 代码#!/usr/bin/env python#coding=utf-8import random#生成[0, 1)直接随机浮点数print
- 说明1、导入unittest模块。2、导入被测对象。3、创建测试类unittest.TestCase。4、重写setUp和tearDown(
- 一、简介pd.concat()函数可以沿着指定的轴将多个dataframe或者series拼接到一起。基本语法:pd.concat( obj
- 目录简介时间分类TimestampDatetimeIndexdate_range 和 bdate_rangeorigin格式化PeriodD
- 存储过程的优缺点: 存储过程优点: 1.由于应用程序随着时间推移会不断更改,增删功能,T-SQL过程代码会变得更复杂,StoredProce
- 最近做的一个项目对日期时间的处理比较多,最后整理到一个工具类里面,方便以后使用:1. 在utils文件夹下新建一个dateTimeUtil.
- 正则表达式,贪婪匹配与非贪婪匹配正则表达式前戏以某app注册页面获取手机号为例. 其有很多校验规则: 国内手机号必须是11位,纯数字,是常规