中文字幕av专区_日韩电影在线播放_精品国产精品久久一区免费式_av在线免费观看网站

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

Django多層嵌套ManyToMany字段ORM的使用方法

發布時間:2020-07-23 15:04:53 來源:億速云 閱讀:263 作者:小豬 欄目:開發技術

這篇文章主要講解了Django多層嵌套ManyToMany字段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的使用方法有進一步的了解,如果還想學習更多內容,歡迎關注億速云行業資訊頻道。

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

阿坝县| 广安市| 普陀区| 克山县| 治县。| 巨野县| 金秀| 汉中市| 凭祥市| 烟台市| 泰来县| 盐边县| 本溪| 松江区| 武鸣县| 旬邑县| 崇礼县| 衡南县| 湄潭县| 塘沽区| 长岛县| 福安市| 荥阳市| 武义县| 吉安市| 高青县| 温州市| 饶河县| 贺兰县| 新源县| 康定县| 荣成市| 新平| 蒲江县| 九江县| 怀宁县| 东海县| 九龙城区| 汉阴县| 仪陇县| 铜川市|