Django多層嵌套ManyToMany字段ORM操作詳解
在用django寫項目時,遇到了許多場景,關(guān)于ORM操作獲取數(shù)據(jù)的,但是不好描述出來,百度搜索關(guān)鍵詞都不知道該怎么搜,導(dǎo)致一個人鼓搗了好久。這里細化下問題,還原場景,記錄踩下的坑
首先先列舉model,我舉些生活中的例子,更方便理解問題
# 習(xí)題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='學(xué)習(xí)進度', default=0)
假設(shè)是一本數(shù)學(xué)書,有5個章節(jié),每個章節(jié)里有數(shù)量不等的習(xí)題,
即book與chapter是多對多,chapter與problem也是多對多
場景一: 書籍下的所有習(xí)題
# 按我的理解是取問題非空的章節(jié)數(shù)# 類似于問爺爺有幾個孫子,沒辦法跨輩,就按一個孫子對應(yīng)一個爸爸來?。ㄓ兄貜?fù))book.chapter.filter(problem___id__isnull=False).count()
場景二:書籍下所有通過的習(xí)題
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
# 習(xí)題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='學(xué)習(xí)進度', 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設(shè)置了ordering后,使用distinct()和annotate()問題記錄
model類如下,我在class Meta中設(shè)置了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è)務(wù)有需求如對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é)果并不像我們預(yù)期的一樣,之后把執(zhí)行的sql輸出出來可以看到在group by的時候是對host_name和date_create進行分組,原因就是因為我們在model類中設(shè)置了ordering,去掉之后代碼運行正常。
使用distinct和上面的情況類似,就不列出來了。
以上這篇Django多層嵌套ManyToMany字段ORM操作詳解就是小編分享給大家的全部內(nèi)容了,希望能給大家一個參考,也希望大家多多支持好吧啦網(wǎng)。
相關(guān)文章:
1. php網(wǎng)絡(luò)安全中命令執(zhí)行漏洞的產(chǎn)生及本質(zhì)探究2. 三個不常見的 HTML5 實用新特性簡介3. Angular獲取ngIf渲染的Dom元素示例4. IIS+PHP添加對webp格式圖像的支持配置方法5. ASP調(diào)用WebService轉(zhuǎn)化成JSON數(shù)據(jù),附j(luò)son.min.asp6. 無線標記語言(WML)基礎(chǔ)之WMLScript 基礎(chǔ)第1/2頁7. 使用.net core 自帶DI框架實現(xiàn)延遲加載功能8. Warning: require(): open_basedir restriction in effect,目錄配置open_basedir報錯問題分析9. php測試程序運行速度和頁面執(zhí)行速度的代碼10. ASP.NET Core 5.0中的Host.CreateDefaultBuilder執(zhí)行過程解析
