国产成人精品久久免费动漫-国产成人精品天堂-国产成人精品区在线观看-国产成人精品日本-a级毛片无码免费真人-a级毛片毛片免费观看久潮喷

您的位置:首頁(yè)技術(shù)文章
文章詳情頁(yè)

Django多層嵌套ManyToMany字段ORM操作詳解

瀏覽:2日期:2024-10-04 18:22:36

在用django寫(xiě)項(xiàng)目時(shí),遇到了許多場(chǎng)景,關(guān)于ORM操作獲取數(shù)據(jù)的,但是不好描述出來(lái),百度搜索關(guān)鍵詞都不知道該怎么搜,導(dǎo)致一個(gè)人鼓搗了好久。這里細(xì)化下問(wèn)題,還原場(chǎng)景,記錄踩下的坑

首先先列舉model,我舉些生活中的例子,更方便理解問(wèn)題

# 習(xí)題class Problem(models.Model): desc = models.CharField() answer = models.TextField() is_pass = models.BooleanField(default=False, verbose_name='是否通過(guò)')# 章節(jié)class Chapter(models.Model): _id = models.IntegerField(verbose_name='編號(hào)') title = models.CharField() problem = models.ManyToManyField(Problem) pass_rate = models.IntegerField(verbose_name='通關(guān)率')# 書(shū)籍 class Book(models.Model): title = models.CharField() desc = models.TextField() chapter = models.ManyToManyField(Chapter,verbose_name='章節(jié)') speed = models.IntegerField(verbose_name='學(xué)習(xí)進(jìn)度', default=0)

假設(shè)是一本數(shù)學(xué)書(shū),有5個(gè)章節(jié),每個(gè)章節(jié)里有數(shù)量不等的習(xí)題,

即book與chapter是多對(duì)多,chapter與problem也是多對(duì)多

場(chǎng)景一: 書(shū)籍下的所有習(xí)題

# 按我的理解是取問(wèn)題非空的章節(jié)數(shù)# 類(lèi)似于問(wèn)爺爺有幾個(gè)孫子,沒(méi)辦法跨輩,就按一個(gè)孫子對(duì)應(yīng)一個(gè)爸爸來(lái)取(有重復(fù))book.chapter.filter(problem___id__isnull=False).count()

場(chǎng)景二:書(shū)籍下所有通過(guò)的習(xí)題

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

場(chǎng)景三: 判斷某個(gè)問(wèn)題是否在這本書(shū)里

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

盡可能的減少view中對(duì)models的取值操作,所以把上面幾個(gè)場(chǎng)景方法寫(xiě)在models類(lèi)中

最終的models

# 習(xí)題class Problem(models.Model): desc = models.CharField() answer = models.TextField() is_pass = models.BooleanField(default=False, verbose_name='是否通過(guò)')# 章節(jié)class Chapter(models.Model): _id = models.IntegerField(verbose_name='編號(hào)') 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() # 書(shū)籍 class Book(models.Model): title = models.CharField() desc = models.TextField() chapter = models.ManyToManyField(Chapter,verbose_name='章節(jié)') speed = models.IntegerField(verbose_name='學(xué)習(xí)進(jìn)度', 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()

補(bǔ)充知識(shí):django中當(dāng)model設(shè)置了ordering后,使用distinct()和annotate()問(wèn)題記錄

model類(lèi)如下,我在class Meta中設(shè)置了ordering = [’-date_create’],即模型對(duì)象返回的記錄結(jié)果集是按照這個(gè)字段排序的。

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

當(dāng)業(yè)務(wù)有需求如對(duì)host_name進(jìn)行分組顯示,在代碼中用到了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’: ’點(diǎn)2’, ’dcount’: 1}, {’host_name’: ’點(diǎn)3’, ’dcount’: 2}, {’host_name’: ’點(diǎn)2’, ’dcount’: 1}, {’host_name’: ’點(diǎn)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輸出出來(lái)可以看到在group by的時(shí)候是對(duì)host_name和date_create進(jìn)行分組,原因就是因?yàn)槲覀冊(cè)趍odel類(lèi)中設(shè)置了ordering,去掉之后代碼運(yùn)行正常。

使用distinct和上面的情況類(lèi)似,就不列出來(lái)了。

以上這篇Django多層嵌套ManyToMany字段ORM操作詳解就是小編分享給大家的全部?jī)?nèi)容了,希望能給大家一個(gè)參考,也希望大家多多支持好吧啦網(wǎng)。

標(biāo)簽: Django
相關(guān)文章:
主站蜘蛛池模板: 特黄日韩免费一区二区三区 | 免费一区二区三区久久 | 真人一级一级特黄高清毛片 | 中国老太卖淫播放毛片 | 国产亚洲欧美在线视频 | 亚洲欧美在线不卡 | 亚洲国产三级在线观看 | 91久久青青草原线免费 | 国产精品成人久久久久久久 | 一级爱做片免费观看久久 | 成人国产在线看不卡 | 国产精品漂亮美女在线观看 | 杨幂国产精品福利在线观看 | 亚洲精品一区二区三区www | 国产欧美精品一区二区三区四区 | 国产精品7m凸凹视频分类大全 | 91在线成人 | 欧美一级毛片在线观看 | 国产99久久精品 | 天天综合天天看夜夜添狠狠玩 | 欧美成人免费 | 美女个护士一级毛片亚洲 | 国产精品18久久久久久vr | 日韩欧美亚洲每的更新在线 | 国内精品一区二区在线观看 | 成人精品一区二区久久久 | 日韩国产精品欧美一区二区 | 免费一级做a爰片久久毛片 免费一级做a爰片性色毛片 | 久久a热6 | 国产亚洲欧美日韩在线看片 | 性高湖久久久久久久久aaaaa | 欧美高清强视频 | 久久久久国产午夜 | 一级黄一片 | 日韩一级在线播放免费观看 | 免费在线观看亚洲 | 久久综合婷婷香五月 | 青青爽国产手机在线观看免费 | 国产欧美日韩精品在线 | 青青自拍 | 黄色毛片免费 |