有两种原因可能使你想要自定义 Manager
:添加额外的 Manager
方法,修改 Manager
返回的原始 QuerySet
。
添加额外的管理器方法
添加额外的 Manager
方法一般是为模型添加 “表级” 功能的更好方法。(对于 “行级” 功能 —— 即,只操作单个模型对象 —— 通过 模型方法,而不是自定义 Manager
的方法。)
自定义 Manager
方法能返回任何东西,没有强制它必须返回一个 QuerySet
。
例如,这个自定义 Manager
提供了一个方法 with_counts()
,它会返回包含所有 OpinionPoll
对象的列表。每个对象都有一个额外属性 num_response
,这是一次聚合查询的结果:
from django.db import models class PollManager(models.Manager): def with_counts(self): from django.db import connection with connection.cursor() as cursor: cursor.execute(""" SELECT p.id, p.question, p.poll_date, COUNT(*) FROM polls_opinionpoll p, polls_response r WHERE p.id = r.poll_id GROUP BY p.id, p.question, p.poll_date ORDER BY p.poll_date DESC""") result_list = [] for row in cursor.fetchall(): p = self.model(id=row[0], question=row[1], poll_date=row[2]) p.num_responses = row[3] result_list.append(p) return result_list class OpinionPoll(models.Model): question = models.CharField(max_length=200) poll_date = models.DateField() objects = PollManager() class Response(models.Model): poll = models.ForeignKey(OpinionPoll, on_delete=models.CASCADE) person_name = models.CharField(max_length=50) response = models.TextField()
通过本例,你可以利用 OpinionPoll.objects.with_counts()
返回一个包含 OpinionPoll
对象的列表,每个对象都有 num_responses
属性。
本例的另一个注意事项是 Manager
方法能通过 self.model
获取所依附的模型类。
修改管理器的初始 QuerySet
Manager
的基础 QuerySet
会返回系统中所有的对象。例如,使用以下模型:
from django.db import models class Book(models.Model): title = models.CharField(max_length=100) author = models.CharField(max_length=50)
……语句 Book.objects.all()
会返回数据库中所有的书。
你可以通过重写 Manager.get_queryset()
方法来覆盖 Manager
的基础 QuerySet
。 get_queryset()
返回的 QuerySet
应该包含你需要的属性。
例如,以下模型有 两个 Manager
—— 一个返回所有对象,另一个仅返回 Roald Dahl 写的书:
# First, define the Manager subclass. class DahlBookManager(models.Manager): def get_queryset(self): return super().get_queryset().filter(author='Roald Dahl') # Then hook it into the Book model explicitly. class Book(models.Model): title = models.CharField(max_length=100) author = models.CharField(max_length=50) objects = models.Manager() # The default manager. dahl_objects = DahlBookManager() # The Dahl-specific manager.
使用这个实例模型时, Book.objects.all()
会返回数据库中所有的书,而 Book.dahl_objects.all()
仅返回 Roald Dahl 写的书。