详解Python的Django框架中Manager方法的使用


Posted in Python onJuly 21, 2015

在语句Book.objects.all()中,objects是一个特殊的属性,需要通过它查询数据库。 在第5章,我们只是简要地说这是模块的manager 。现在是时候深入了解managers是什么和如何使用了。

总之,模块manager是一个对象,Django模块通过它进行数据库查询。 每个Django模块至少有一个manager,你可以创建自定义manager以定制数据库访问。

下面是你创建自定义manager的两个原因: 增加额外的manager方法,和/或修manager返回的初始QuerySet。
增加额外的Manager方法

增加额外的manager方法是为模块添加表级功能的首选办法。

例如,我们为Book模型定义了一个title_count()方法,它需要一个关键字,返回包含这个关键字的书的数量。 (这个例子有点牵强,不过它可以说明managers如何工作。)

# models.py

from django.db import models

# ... Author and Publisher models here ...

**class BookManager(models.Manager):**
  **def title_count(self, keyword):**
    **return self.filter(title__icontains=keyword).count()**

class Book(models.Model):
  title = models.CharField(max_length=100)
  authors = models.ManyToManyField(Author)
  publisher = models.ForeignKey(Publisher)
  publication_date = models.DateField()
  num_pages = models.IntegerField(blank=True, null=True)
  **objects = BookManager()**

  def __unicode__(self):
    return self.title

有了这个manager,我们现在可以这样做:

>>> Book.objects.title_count('django')
4
>>> Book.objects.title_count('python')
18

下面是编码该注意的一些地方:

  •     我们建立了一个BookManager类,它继承了django.db.models.Manager。这个类只有一个title_count()方法,用来做统计。 注意,这个方法使用了self.filter(),此处self指manager本身。
  •     我们把BookManager()赋值给模型的objects属性。 它将取代模型的默认manager(objects)如果我们没有特别定义,它将会被自动创建。 我们把它命名为objects,这是为了与自动创建的manager保持一致。

为什么我们要添加一个title_count()方法呢?是为了将经常使用的查询进行封装,这样我们就不必重复编码了。
修改初始Manager QuerySets

manager的基本QuerySet返回系统中的所有对象。 例如,`` Book.objects.all()`` 返回数据库book中的所有书本。

我们可以通过覆盖Manager.get_query_set()方法来重写manager的基本QuerySet。 get_query_set()按照你的要求返回一个QuerySet。

例如,下面的模型有* 两个* manager。一个返回所有对像,另一个只返回作者是Roald Dahl的书。

from django.db import models

**# First, define the Manager subclass.**
**class DahlBookManager(models.Manager):**
  **def get_query_set(self):**
    **return super(DahlBookManager, self).get_query_set().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()只返回了一本. 注意我们明确地将objects设置成manager的实例,因为如果我们不这么做,那么唯一可用的manager就将是dah1_objects。

当然,由于get_query_set()返回的是一个QuerySet对象,所以我们可以使用filter(),exclude()和其他一切QuerySet的方法。 像这些语法都是正确的:

Book.dahl_objects.all()
Book.dahl_objects.filter(title='Matilda')
Book.dahl_objects.count()

这个例子也指出了其他有趣的技术: 在同一个模型中使用多个manager。 只要你愿意,你可以为你的模型添加多个manager()实例。 这是一个为模型添加通用滤器的简单方法。

例如:

class MaleManager(models.Manager):
  def get_query_set(self):
    return super(MaleManager, self).get_query_set().filter(sex='M')

class FemaleManager(models.Manager):
  def get_query_set(self):
    return super(FemaleManager, self).get_query_set().filter(sex='F')

class Person(models.Model):
  first_name = models.CharField(max_length=50)
  last_name = models.CharField(max_length=50)
  sex = models.CharField(max_length=1, choices=(('M', 'Male'), ('F', 'Female')))
  people = models.Manager()
  men = MaleManager()
  women = FemaleManager()

这个例子允许你执行`` Person.men.all()`` ,`` Person.women.all()`` ,`` Person.people.all()`` 查询,生成你想要的结果。

如果你使用自定义的Manager对象,请注意,Django遇到的第一个Manager(以它在模型中被定义的位置为准)会有一个特殊状态。 Django将会把第一个Manager 定义为默认Manager ,Django的许多部分(但是不包括admin应用)将会明确地为模型使用这个manager。 结论是,你应该小心地选择你的默认manager。因为覆盖get_query_set() 了,你可能接受到一个无用的返回对像,你必须避免这种情况。

Python 相关文章推荐
Python程序设计入门(2)变量类型简介
Jun 16 Python
Python3.2模拟实现webqq登录
Feb 15 Python
Python爬取网易云音乐热门评论
Mar 31 Python
Python入门_条件控制(详解)
May 16 Python
Python Nose框架编写测试用例方法
Oct 26 Python
Django 跨域请求处理的示例代码
May 02 Python
python实现图片中文字分割效果
Jul 22 Python
Python数据分析pandas模块用法实例详解
Nov 20 Python
tensorflow 保存模型和取出中间权重例子
Jan 24 Python
python 两个一样的字符串用==结果为false问题的解决
Mar 12 Python
教你使用Sublime text3搭建Python开发环境及常用插件安装另分享Sublime text3最新激活注册码
Nov 12 Python
python matplotlib工具栏源码探析三之添加、删除自定义工具项的案例详解
Feb 25 Python
通过数据库对Django进行删除字段和删除模型的操作
Jul 21 #Python
通过数据库向Django模型添加字段的示例
Jul 21 #Python
Django的数据模型访问多对多键值的方法
Jul 21 #Python
举例讲解Django中数据模型访问外键值的方法
Jul 21 #Python
编写自定义的Django模板加载器的简单示例
Jul 21 #Python
详解Python的Django框架中inclusion_tag的使用
Jul 21 #Python
剖析Django中模版标签的解析与参数传递
Jul 21 #Python
You might like
DC动漫人物排行
2020/03/03 欧美动漫
浅谈PHP eval()函数定义和用法
2016/06/21 PHP
详解PHP中的8个魔术常量
2020/07/06 PHP
PHP开发者必须掌握的6个关键字
2014/04/14 Javascript
jquery根据锚点offset值实现动画切换
2014/09/11 Javascript
Jquery实现瀑布流布局(备有详细注释)
2015/07/31 Javascript
Javascript连接Access数据库完整实例
2015/08/03 Javascript
利用AJAX实现WordPress中的文章列表及评论的分页功能
2016/05/17 Javascript
JS仿JQuery选择器功能
2017/03/08 Javascript
jQuery插件HighCharts实现的2D面积图效果示例【附demo源码下载】
2017/03/15 Javascript
详解angular中通过$location获取路径(参数)的写法
2017/03/21 Javascript
详解vue-router基本使用
2017/04/18 Javascript
vue实现全选和反选功能
2017/08/31 Javascript
Vue中this.$router.push参数获取方法
2018/02/27 Javascript
javaScript实现鼠标在文字上悬浮时弹出悬浮层效果
2020/04/12 Javascript
JS实现的汉字与Unicode码相互转化功能分析
2018/05/25 Javascript
详解使用Next.js构建服务端渲染应用
2018/07/10 Javascript
vue解决一个方法同时发送多个请求的问题
2018/09/25 Javascript
利用React Router4实现的服务端直出渲染(SSR)
2019/01/07 Javascript
[08:44]DOTA2发布会群星聚首 我们都是刀塔人
2014/03/21 DOTA
人脸识别经典算法一 特征脸方法(Eigenface)
2018/03/13 Python
tensorflow2.0与tensorflow1.0的性能区别介绍
2020/02/07 Python
python程序需要编译吗
2020/06/19 Python
python绘图pyecharts+pandas的使用详解
2020/12/13 Python
CSS3 Flexbox中flex-shrink属性的用法示例介绍
2013/12/30 HTML / CSS
HTML5的postMessage的使用手册
2018/12/19 HTML / CSS
Reebok俄罗斯官方网上商店:购买锐步运动服装和鞋子
2016/09/26 全球购物
澳大利亚宠物食品和药物在线:Jumbo Pets
2018/03/24 全球购物
AssertionError 跟一下那个类是 “is – a”的关系
2012/02/21 面试题
影视后期实训报告
2014/11/05 职场文书
2014年团委工作总结
2014/11/13 职场文书
拉贝日记观后感
2015/06/05 职场文书
2016年暑期教师培训心得体会
2016/01/09 职场文书
SQL Server中交叉联接的用法详解
2021/04/22 SQL Server
PyTorch dropout设置训练和测试模式的实现
2021/05/27 Python
golang操作redis的客户端包有多个比如redigo、go-redis
2022/04/14 Golang