Django多層嵌套ManyToMany字段ORM操作詳解
在用django寫項目時,遇到了許多場景,關(guān)于ORM操作獲取數(shù)據(jù)的,但是不好描述出來,百度搜索關(guān)鍵詞都不知道該怎么搜,導致一個人鼓搗了好久。這里細化下問題,還原場景,記錄踩下的坑
首先先列舉model,我舉些生活中的例子,更方便理解問題
# 習題 class Problem(models.Model): desc = models.CharField() answer = models.TextField() is_pass = models.BooleanField(default=False, verbose_name="是否通過") # 章節(jié) class Chapter(models.Model): _id = models.IntegerField(verbose_name="編號") title = models.CharField() problem = models.ManyToManyField(Problem) pass_rate = models.IntegerField(verbose_name="通關(guān)率") # 書籍 class Book(models.Model): title = models.CharField() desc = models.TextField() chapter = models.ManyToManyField(Chapter,verbose_name="章節(jié)") speed = models.IntegerField(verbose_name="學習進度", default=0)
假設是一本數(shù)學書,有5個章節(jié),每個章節(jié)里有數(shù)量不等的習題,
即book與chapter是多對多,chapter與problem也是多對多
場景一: 書籍下的所有習題
# 按我的理解是取問題非空的章節(jié)數(shù) # 類似于問爺爺有幾個孫子,沒辦法跨輩,就按一個孫子對應一個爸爸來?。ㄓ兄貜停? 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="是否通過")
# 章節(jié)
class Chapter(models.Model):
_id = models.IntegerField(verbose_name="編號")
title = models.CharField()
problem = models.ManyToManyField(Problem)
pass_rate = models.IntegerField(verbose_name="通關(guān)率")
@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="章節(jié)")
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'],即模型對象返回的記錄結(jié)果集是按照這個字段排序的。
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
當業(yè)務有需求如對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
可以看到,所得到的結(jié)果并不像我們預期的一樣,之后把執(zhí)行的sql輸出出來可以看到在group by的時候是對host_name和date_create進行分組,原因就是因為我們在model類中設置了ordering,去掉之后代碼運行正常。
使用distinct和上面的情況類似,就不列出來了。
以上這篇Django多層嵌套ManyToMany字段ORM操作詳解就是小編分享給大家的全部內(nèi)容了,希望能給大家一個參考,也希望大家多多支持腳本之家。
相關(guān)文章
SpringBoot實現(xiàn)登錄注冊常見問題解決方案
這篇文章主要介紹了SpringBoot實現(xiàn)登錄注冊常見問題解決方案,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友可以參考下2020-03-03
Python模塊結(jié)構(gòu)與布局操作方法實例分析
這篇文章主要介紹了Python模塊結(jié)構(gòu)與布局操作方法,結(jié)合實例形式分析了Python模塊與布局的相關(guān)概念、使用方法與相關(guān)注意事項,需要的朋友可以參考下2017-07-07
django自帶的server 讓外網(wǎng)主機訪問方法
今天小編就為大家分享一篇django自帶的server 讓外網(wǎng)主機訪問方法。具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2018-05-05
網(wǎng)絡瀏覽器中運行Python腳本PyScript剖析
這篇文章主要為大家介紹了網(wǎng)絡瀏覽器中運行Python腳本PyScript剖析詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2022-08-08
python爬蟲使用正則爬取網(wǎng)站的實現(xiàn)
這篇文章主要介紹了python爬蟲使用正則爬取網(wǎng)站的實現(xiàn),文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧2020-08-08

